#!/usr/bin/env python # # Copyright (C) 2016 GNS3 Technologies Inc. # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . import os import uuid import shutil import tempfile import asyncio import aiohttp.web from datetime import datetime, timezone from ..utils.asyncio import wait_run_in_executor from .export_project import export_project from .import_project import import_project # The string use to extract the date from the filename FILENAME_TIME_FORMAT = "%d%m%y_%H%M%S" class Snapshot: """ A snapshot object """ def __init__(self, project, name=None, filename=None): assert filename or name, "You need to pass a name or a filename" self._id = str(uuid.uuid4()) # We don't need to keep id between project loading because they are use only as key for operation like delete, update.. but have no impact on disk self._project = project if name: self._name = name self._created_at = datetime.now().timestamp() filename = self._name + "_" + datetime.utcfromtimestamp(self._created_at).replace(tzinfo=None).strftime(FILENAME_TIME_FORMAT) + ".gns3project" else: self._name = filename.split("_")[0] datestring = filename.replace(self._name + "_", "").split(".")[0] try: self._created_at = datetime.strptime(datestring, FILENAME_TIME_FORMAT).replace(tzinfo=timezone.utc).timestamp() except ValueError: self._created_at = datetime.utcnow().timestamp() self._path = os.path.join(project.path, "snapshots", filename) @property def id(self): return self._id @property def name(self): return self._name @property def path(self): return self._path @property def created_at(self): return int(self._created_at) def _create_snapshot_file(self, zipstream): """ Creates the snapshot file (to be run in its own thread) """ with open(self.path, "wb") as f: for data in zipstream: f.write(data) async def create(self): """ Create the snapshot """ if os.path.exists(self.path): raise aiohttp.web.HTTPConflict(text="The snapshot file '{}' already exists".format(self.name)) snapshot_directory = os.path.join(self._project.path, "snapshots") try: os.makedirs(snapshot_directory, exist_ok=True) except OSError as e: raise aiohttp.web.HTTPInternalServerError(text="Could not create the snapshot directory '{}': {}".format(snapshot_directory, e)) try: with tempfile.TemporaryDirectory() as tmpdir: zipstream = await export_project(self._project, tmpdir, keep_compute_id=True, allow_all_nodes=True) await wait_run_in_executor(self._create_snapshot_file, zipstream) except (ValueError, OSError, RuntimeError) as e: raise aiohttp.web.HTTPConflict(text="Could not create snapshot file '{}': {}".format(self.path, e)) async def restore(self): """ Restore the snapshot """ await self._project.delete_on_computes() # We don't send close notification to clients because the close / open dance is purely internal await self._project.close(ignore_notification=True) try: # delete the current project files project_files_path = os.path.join(self._project.path, "project-files") if os.path.exists(project_files_path): await wait_run_in_executor(shutil.rmtree, project_files_path) with open(self._path, "rb") as f: project = await import_project(self._project.controller, self._project.id, f, location=self._project.path) except (OSError, PermissionError) as e: raise aiohttp.web.HTTPConflict(text=str(e)) await project.open() self._project.emit_notification("snapshot.restored", self.__json__()) return self._project def __json__(self): return { "snapshot_id": self._id, "name": self._name, "created_at": int(self._created_at), "project_id": self._project.id }