1
0
mirror of https://github.com/GNS3/gns3-server synced 2025-01-12 09:00:57 +00:00

Drop code for temporary projects

Fix https://github.com/GNS3/gns3-gui/issues/982
This commit is contained in:
Julien Duponchelle 2016-05-24 17:54:08 +02:00
parent df73f80bf5
commit a797038aeb
No known key found for this signature in database
GPG Key ID: CE8B29639E07F5E8
12 changed files with 25 additions and 224 deletions

View File

@ -44,10 +44,9 @@ class Project:
:param project_id: force project identifier (None by default auto generate an UUID) :param project_id: force project identifier (None by default auto generate an UUID)
:param path: path of the project. (None use the standard directory) :param path: path of the project. (None use the standard directory)
:param temporary: boolean to tell if the project is a temporary project (destroy when closed)
""" """
def __init__(self, name=None, project_id=None, path=None, temporary=False): def __init__(self, name=None, project_id=None, path=None):
self._name = name self._name = name
try: try:
@ -58,7 +57,6 @@ class Project:
self._nodes = set() self._nodes = set()
self._nodes_to_destroy = set() self._nodes_to_destroy = set()
self.temporary = temporary
self._used_tcp_ports = set() self._used_tcp_ports = set()
self._used_udp_ports = set() self._used_udp_ports = set()
@ -83,8 +81,7 @@ class Project:
return { return {
"name": self._name, "name": self._name,
"project_id": self._id, "project_id": self._id
"temporary": self._temporary
} }
def _config(self): def _config(self):
@ -114,19 +111,6 @@ class Project:
raise aiohttp.web.HTTPForbidden(text="You are not allowed to modify the project directory path") raise aiohttp.web.HTTPForbidden(text="You are not allowed to modify the project directory path")
self._path = path self._path = path
self._update_temporary_file()
@asyncio.coroutine
def clean_old_path(self, old_path):
"""
Called after a project location change. All the compute should
have been notified before
"""
if self._temporary:
try:
yield from wait_run_in_executor(shutil.rmtree, old_path)
except OSError as e:
log.warn("Can't remove temporary directory {}: {}".format(old_path, e))
@property @property
def name(self): def name(self):
@ -145,20 +129,6 @@ class Project:
return self._nodes return self._nodes
@property
def temporary(self):
return self._temporary
@temporary.setter
def temporary(self, temporary):
if hasattr(self, 'temporary') and temporary == self._temporary:
return
self._temporary = temporary
self._update_temporary_file()
def record_tcp_port(self, port): def record_tcp_port(self, port):
""" """
Associate a reserved TCP port number with this project. Associate a reserved TCP port number with this project.
@ -199,27 +169,6 @@ class Project:
if port in self._used_udp_ports: if port in self._used_udp_ports:
self._used_udp_ports.remove(port) self._used_udp_ports.remove(port)
def _update_temporary_file(self):
"""
Update the .gns3_temporary file in order to reflect the current project status.
"""
if not hasattr(self, "_path"):
return
if self._temporary:
try:
with open(os.path.join(self._path, ".gns3_temporary"), 'w+') as f:
f.write("1")
except OSError as e:
raise aiohttp.web.HTTPInternalServerError(text="Could not create temporary project: {}".format(e))
else:
if os.path.exists(os.path.join(self._path, ".gns3_temporary")):
try:
os.remove(os.path.join(self._path, ".gns3_temporary"))
except OSError as e:
raise aiohttp.web.HTTPInternalServerError(text="Could not mark project as no longer temporary: {}".format(e))
def module_working_directory(self, module_name): def module_working_directory(self, module_name):
""" """
Returns a working directory for the module Returns a working directory for the module
@ -319,7 +268,7 @@ class Project:
for module in self.compute(): for module in self.compute():
yield from module.instance().project_closing(self) yield from module.instance().project_closing(self)
yield from self._close_and_clean(self._temporary) yield from self._close_and_clean(False)
for module in self.compute(): for module in self.compute():
yield from module.instance().project_closed(self) yield from module.instance().project_closed(self)
@ -395,20 +344,6 @@ class Project:
for module in self.compute(): for module in self.compute():
yield from module.instance().project_closed(self) yield from module.instance().project_closed(self)
@classmethod
def clean_project_directory(cls):
"""
At startup drop old temporary project. After a crash for example
"""
directory = get_default_project_directory()
if os.path.exists(directory):
for project in os.listdir(directory):
path = os.path.join(directory, project)
if os.path.exists(os.path.join(path, ".gns3_temporary")):
log.warning("Purge old temporary project {}".format(project))
shutil.rmtree(path)
def compute(self): def compute(self):
""" """
Returns all loaded modules from compute. Returns all loaded modules from compute.

View File

@ -70,7 +70,7 @@ class ProjectManager:
raise aiohttp.web.HTTPNotFound(text="Project ID {} doesn't exist".format(project_id)) raise aiohttp.web.HTTPNotFound(text="Project ID {} doesn't exist".format(project_id))
return self._projects[project_id] return self._projects[project_id]
def create_project(self, name=None, project_id=None, path=None, temporary=False): def create_project(self, name=None, project_id=None, path=None):
""" """
Create a project and keep a references to it in project manager. Create a project and keep a references to it in project manager.
@ -79,7 +79,7 @@ class ProjectManager:
if project_id is not None and project_id in self._projects: if project_id is not None and project_id in self._projects:
return self._projects[project_id] return self._projects[project_id]
project = Project(name=name, project_id=project_id, path=path, temporary=temporary) project = Project(name=name, project_id=project_id, path=path)
self._projects[project.id] = project self._projects[project.id] = project
return project return project

View File

@ -46,16 +46,14 @@ class Controller:
self._config_file = os.path.join(config_path, "gns3_controller.conf") self._config_file = os.path.join(config_path, "gns3_controller.conf")
server_config = Config.instance().get_section_config("Server") server_config = Config.instance().get_section_config("Server")
print(server_config)
if server_config.getboolean("local", False) is True: if server_config.getboolean("local", False) is True:
print("MMOOOOOOONKEYYYYYY")
self._computes["local"] = Compute(compute_id="local", self._computes["local"] = Compute(compute_id="local",
controller=self, controller=self,
protocol=server_config.get("protocol", "http"), protocol=server_config.get("protocol", "http"),
host=server_config.get("host", "localhost"), host=server_config.get("host", "localhost"),
port=server_config.getint("port", 3080), port=server_config.getint("port", 3080),
user=server_config.get("user", ""), user=server_config.get("user", ""),
password=server_config.get("password", "")) password=server_config.get("password", ""))
def save(self): def save(self):
""" """
@ -107,7 +105,6 @@ class Controller:
""" """
if compute_id not in self._computes: if compute_id not in self._computes:
# We disallow to create from the outside the # We disallow to create from the outside the
if compute_id == 'local': if compute_id == 'local':
return None return None

View File

@ -34,10 +34,9 @@ class Project:
:param project_id: force project identifier (None by default auto generate an UUID) :param project_id: force project identifier (None by default auto generate an UUID)
:param path: path of the project. (None use the standard directory) :param path: path of the project. (None use the standard directory)
:param temporary: boolean to tell if the project is a temporary project (destroy when closed)
""" """
def __init__(self, name=None, project_id=None, path=None, temporary=False, controller=None): def __init__(self, name=None, project_id=None, path=None, controller=None):
self._controller = controller self._controller = controller
self._name = name self._name = name
@ -54,7 +53,6 @@ class Project:
path = os.path.join(get_default_project_directory(), self._id) path = os.path.join(get_default_project_directory(), self._id)
self.path = path self.path = path
self._temporary = temporary
self._computes = set() self._computes = set()
self._nodes = {} self._nodes = {}
self._links = {} self._links = {}
@ -74,10 +72,6 @@ class Project:
def id(self): def id(self):
return self._id return self._id
@property
def temporary(self):
return self._temporary
@property @property
def path(self): def path(self):
return self._path return self._path
@ -126,14 +120,12 @@ class Project:
yield from compute.post("/projects", data={ yield from compute.post("/projects", data={
"name": self._name, "name": self._name,
"project_id": self._id, "project_id": self._id,
"temporary": self._temporary,
"path": self._path "path": self._path
}) })
else: else:
yield from compute.post("/projects", data={ yield from compute.post("/projects", data={
"name": self._name, "name": self._name,
"project_id": self._id, "project_id": self._id,
"temporary": self._temporary
}) })
self._project_created_on_compute.add(compute) self._project_created_on_compute.add(compute)
@ -235,6 +227,5 @@ class Project:
return { return {
"name": self._name, "name": self._name,
"project_id": self._id, "project_id": self._id,
"temporary": self._temporary,
"path": self._path "path": self._path
} }

View File

@ -73,8 +73,7 @@ class ProjectHandler:
p = pm.create_project( p = pm.create_project(
name=request.json.get("name"), name=request.json.get("name"),
path=request.json.get("path"), path=request.json.get("path"),
project_id=request.json.get("project_id"), project_id=request.json.get("project_id")
temporary=request.json.get("temporary", False)
) )
response.set_status(201) response.set_status(201)
response.json(p) response.json(p)
@ -120,9 +119,6 @@ class ProjectHandler:
project.path = project_path project.path = project_path
for module in MODULES: for module in MODULES:
yield from module.instance().project_moved(project) yield from module.instance().project_moved(project)
yield from project.clean_old_path(old_path)
# Very important: we need to remove temporary flag after moving the project
project.temporary = request.json.get("temporary", project.temporary)
response.json(project) response.json(project)
@Route.post( @Route.post(

View File

@ -47,8 +47,7 @@ class ProjectHandler:
controller = Controller.instance() controller = Controller.instance()
project = yield from controller.add_project(name=request.json.get("name"), project = yield from controller.add_project(name=request.json.get("name"),
path=request.json.get("path"), path=request.json.get("path"),
project_id=request.json.get("project_id"), project_id=request.json.get("project_id"))
temporary=request.json.get("temporary", False))
response.set_status(201) response.set_status(201)
response.json(project) response.json(project)

View File

@ -31,7 +31,6 @@ from gns3server.web.web_server import WebServer
from gns3server.web.logger import init_logger from gns3server.web.logger import init_logger
from gns3server.version import __version__ from gns3server.version import __version__
from gns3server.config import Config from gns3server.config import Config
from gns3server.compute.project import Project
from gns3server.crash_report import CrashReport from gns3server.crash_report import CrashReport
@ -224,8 +223,6 @@ def run():
log.critical("The current working directory doesn't exist") log.critical("The current working directory doesn't exist")
return return
Project.clean_project_directory()
CrashReport.instance() CrashReport.instance()
host = server_config["host"] host = server_config["host"]
port = int(server_config["port"]) port = int(server_config["port"])

View File

@ -37,11 +37,7 @@ PROJECT_CREATE_SCHEMA = {
"minLength": 36, "minLength": 36,
"maxLength": 36, "maxLength": 36,
"pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$" "pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$"
}, }
"temporary": {
"description": "Whether the project is a temporary project or not",
"type": "boolean"
},
}, },
"additionalProperties": False, "additionalProperties": False,
} }
@ -56,10 +52,6 @@ PROJECT_UPDATE_SCHEMA = {
"type": ["string", "null"], "type": ["string", "null"],
"minLength": 1 "minLength": 1
}, },
"temporary": {
"description": "Whether the project is a temporary project or not",
"type": "boolean"
},
"path": { "path": {
"description": "Path of the project on the server (work only with --local)", "description": "Path of the project on the server (work only with --local)",
"type": ["string", "null"] "type": ["string", "null"]
@ -85,10 +77,6 @@ PROJECT_OBJECT_SCHEMA = {
"maxLength": 36, "maxLength": 36,
"pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$" "pattern": "^[a-fA-F0-9]{8}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{4}-[a-fA-F0-9]{12}$"
}, },
"temporary": {
"description": "Whether the project is a temporary project or not",
"type": "boolean"
},
"path": { "path": {
"description": "Project directory", "description": "Project directory",
"type": ["string", "null"], "type": ["string", "null"],
@ -96,7 +84,7 @@ PROJECT_OBJECT_SCHEMA = {
} }
}, },
"additionalProperties": False, "additionalProperties": False,
"required": ["project_id", "temporary"] "required": ["project_id"]
} }
PROJECT_LIST_SCHEMA = { PROJECT_LIST_SCHEMA = {

View File

@ -76,7 +76,6 @@ def test_path(tmpdir):
p = Project(project_id=str(uuid4())) p = Project(project_id=str(uuid4()))
assert p.path == os.path.join(directory, p.id) assert p.path == os.path.join(directory, p.id)
assert os.path.exists(os.path.join(directory, p.id)) assert os.path.exists(os.path.join(directory, p.id))
assert not os.path.exists(os.path.join(p.path, ".gns3_temporary"))
def test_init_path(tmpdir): def test_init_path(tmpdir):
@ -86,31 +85,6 @@ def test_init_path(tmpdir):
assert p.path == str(tmpdir) assert p.path == str(tmpdir)
def test_changing_path_temporary_flag(tmpdir):
with patch("gns3server.compute.project.Project.is_local", return_value=True):
p = Project(temporary=True, project_id=str(uuid4()))
assert os.path.exists(p.path)
original_path = p.path
assert os.path.exists(os.path.join(p.path, ".gns3_temporary"))
p.path = str(tmpdir)
def test_temporary_path():
p = Project(temporary=True, project_id=str(uuid4()))
assert os.path.exists(p.path)
assert os.path.exists(os.path.join(p.path, ".gns3_temporary"))
def test_remove_temporary_flag():
p = Project(temporary=True, project_id=str(uuid4()))
assert os.path.exists(p.path)
assert os.path.exists(os.path.join(p.path, ".gns3_temporary"))
p.temporary = False
assert not os.path.exists(os.path.join(p.path, ".gns3_temporary"))
def test_changing_path_not_allowed(tmpdir): def test_changing_path_not_allowed(tmpdir):
with patch("gns3server.compute.project.Project.is_local", return_value=False): with patch("gns3server.compute.project.Project.is_local", return_value=False):
with pytest.raises(aiohttp.web.HTTPForbidden): with pytest.raises(aiohttp.web.HTTPForbidden):
@ -120,7 +94,7 @@ def test_changing_path_not_allowed(tmpdir):
def test_json(tmpdir): def test_json(tmpdir):
p = Project(project_id=str(uuid4())) p = Project(project_id=str(uuid4()))
assert p.__json__() == {"name": p.name, "project_id": p.id, "temporary": False} assert p.__json__() == {"name": p.name, "project_id": p.id}
def test_node_working_directory(tmpdir, node): def test_node_working_directory(tmpdir, node):
@ -201,40 +175,6 @@ def test_project_close(loop, node, project):
assert node.id not in node.manager._nodes assert node.id not in node.manager._nodes
def test_project_close_temporary_project(loop, manager):
"""A temporary project is deleted when closed"""
project = Project(temporary=True, project_id=str(uuid4()))
directory = project.path
assert os.path.exists(directory)
loop.run_until_complete(asyncio.async(project.close()))
assert os.path.exists(directory) is False
def test_clean_project_directory(tmpdir):
# A non anonymous project with uuid.
project1 = tmpdir / uuid4()
project1.mkdir()
# A non anonymous project.
oldproject = tmpdir / uuid4()
oldproject.mkdir()
# an anonymous project
project2 = tmpdir / uuid4()
project2.mkdir()
tmp = (project2 / ".gns3_temporary")
with open(str(tmp), 'w+') as f:
f.write("1")
with patch("gns3server.config.Config.get_section_config", return_value={"projects_path": str(tmpdir)}):
Project.clean_project_directory()
assert os.path.exists(str(project1))
assert os.path.exists(str(oldproject))
assert not os.path.exists(str(project2))
def test_list_files(tmpdir, loop): def test_list_files(tmpdir, loop):
with patch("gns3server.config.Config.get_section_config", return_value={"projects_path": str(tmpdir)}): with patch("gns3server.config.Config.get_section_config", return_value={"projects_path": str(tmpdir)}):

View File

@ -27,9 +27,10 @@ from uuid import uuid4
from gns3server.controller.project import Project from gns3server.controller.project import Project
from gns3server.config import Config from gns3server.config import Config
@pytest.fixture @pytest.fixture
def project(controller): def project(controller):
return Project(controller=controller) return Project(controller=controller)
def test_affect_uuid(): def test_affect_uuid():
@ -42,7 +43,7 @@ def test_affect_uuid():
def test_json(tmpdir): def test_json(tmpdir):
p = Project() p = Project()
assert p.__json__() == {"name": p.name, "project_id": p.id, "temporary": False, "path": p.path} assert p.__json__() == {"name": p.name, "project_id": p.id, "path": p.path}
def test_path(tmpdir): def test_path(tmpdir):
@ -99,7 +100,6 @@ def test_add_node_local(async_run, controller):
compute.post.assert_any_call('/projects', data={ compute.post.assert_any_call('/projects', data={
"name": project._name, "name": project._name,
"project_id": project._id, "project_id": project._id,
"temporary": project._temporary,
"path": project._path "path": project._path
}) })
compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id), compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id),
@ -127,8 +127,7 @@ def test_add_node_non_local(async_run, controller):
compute.post.assert_any_call('/projects', data={ compute.post.assert_any_call('/projects', data={
"name": project._name, "name": project._name,
"project_id": project._id, "project_id": project._id
"temporary": project._temporary
}) })
compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id), compute.post.assert_any_call('/projects/{}/vpcs/nodes'.format(project.id),
data={'node_id': node.id, data={'node_id': node.id,

View File

@ -44,16 +44,6 @@ def test_create_project_without_dir(http_compute):
response = http_compute.post("/projects", query, example=True) response = http_compute.post("/projects", query, example=True)
assert response.status == 201 assert response.status == 201
assert response.json["project_id"] == "10010203-0405-0607-0809-0a0b0c0d0e0f" assert response.json["project_id"] == "10010203-0405-0607-0809-0a0b0c0d0e0f"
assert response.json["temporary"] is False
assert response.json["name"] == "test"
def test_create_temporary_project(http_compute):
query = {"name": "test", "temporary": True, "project_id": "20010203-0405-0607-0809-0a0b0c0d0e0f"}
response = http_compute.post("/projects", query)
assert response.status == 201
assert response.json["project_id"] == "20010203-0405-0607-0809-0a0b0c0d0e0f"
assert response.json["temporary"] is True
assert response.json["name"] == "test" assert response.json["name"] == "test"
@ -66,13 +56,12 @@ def test_create_project_with_uuid(http_compute):
def test_show_project(http_compute): def test_show_project(http_compute):
query = {"name": "test", "project_id": "40010203-0405-0607-0809-0a0b0c0d0e02", "temporary": False} query = {"name": "test", "project_id": "40010203-0405-0607-0809-0a0b0c0d0e02"}
response = http_compute.post("/projects", query) response = http_compute.post("/projects", query)
assert response.status == 201 assert response.status == 201
response = http_compute.get("/projects/40010203-0405-0607-0809-0a0b0c0d0e02", example=True) response = http_compute.get("/projects/40010203-0405-0607-0809-0a0b0c0d0e02", example=True)
assert len(response.json.keys()) == 3 assert len(response.json.keys()) == 2
assert response.json["project_id"] == "40010203-0405-0607-0809-0a0b0c0d0e02" assert response.json["project_id"] == "40010203-0405-0607-0809-0a0b0c0d0e02"
assert response.json["temporary"] is False
assert response.json["name"] == "test" assert response.json["name"] == "test"
@ -97,35 +86,7 @@ def test_list_projects(http_compute):
assert "51010203-0405-0607-0809-0a0b0c0d0e0f" in [p["project_id"] for p in response.json] assert "51010203-0405-0607-0809-0a0b0c0d0e0f" in [p["project_id"] for p in response.json]
def test_update_temporary_project(http_compute): def test_update_path_project(http_compute, tmpdir):
query = {"name": "test", "temporary": True, "project_id": "60010203-0405-0607-0809-0a0b0c0d0e0b"}
response = http_compute.post("/projects", query)
assert response.status == 201
query = {"name": "test", "temporary": False}
response = http_compute.put("/projects/{project_id}".format(project_id=response.json["project_id"]), query, example=True)
assert response.status == 200
assert response.json["temporary"] is False
def test_update_path_project_temporary(http_compute, tmpdir):
os.makedirs(str(tmpdir / "a"))
os.makedirs(str(tmpdir / "b"))
with patch("gns3server.compute.project.Project.is_local", return_value=True):
response = http_compute.post("/projects", {"name": "first_name", "path": str(tmpdir / "a"), "temporary": True, "project_id": "70010203-0405-0607-0809-0a0b0c0d0e0b"})
assert response.status == 201
assert response.json["name"] == "first_name"
query = {"name": "second_name", "path": str(tmpdir / "b")}
response = http_compute.put("/projects/{project_id}".format(project_id=response.json["project_id"]), query, example=True)
assert response.status == 200
assert response.json["name"] == "second_name"
assert not os.path.exists(str(tmpdir / "a"))
assert os.path.exists(str(tmpdir / "b"))
def test_update_path_project_non_temporary(http_compute, tmpdir):
os.makedirs(str(tmpdir / "a")) os.makedirs(str(tmpdir / "a"))
os.makedirs(str(tmpdir / "b")) os.makedirs(str(tmpdir / "b"))

View File

@ -54,16 +54,14 @@ def test_create_project_without_dir(http_controller):
response = http_controller.post("/projects", query, example=True) response = http_controller.post("/projects", query, example=True)
assert response.status == 201 assert response.status == 201
assert response.json["project_id"] == "10010203-0405-0607-0809-0a0b0c0d0e0f" assert response.json["project_id"] == "10010203-0405-0607-0809-0a0b0c0d0e0f"
assert response.json["temporary"] is False
assert response.json["name"] == "test" assert response.json["name"] == "test"
def test_create_temporary_project(http_controller): def test_create_temporary_project(http_controller):
query = {"name": "test", "temporary": True, "project_id": "20010203-0405-0607-0809-0a0b0c0d0e0f"} query = {"name": "test", "project_id": "20010203-0405-0607-0809-0a0b0c0d0e0f"}
response = http_controller.post("/projects", query) response = http_controller.post("/projects", query)
assert response.status == 201 assert response.status == 201
assert response.json["project_id"] == "20010203-0405-0607-0809-0a0b0c0d0e0f" assert response.json["project_id"] == "20010203-0405-0607-0809-0a0b0c0d0e0f"
assert response.json["temporary"] is True
assert response.json["name"] == "test" assert response.json["name"] == "test"