2016-07-21 18:15:35 +02:00
#!/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 <http://www.gnu.org/licenses/>.
import os
2018-08-25 18:10:40 +07:00
import sys
2016-07-21 18:15:35 +02:00
import json
2016-07-21 20:17:36 +02:00
import asyncio
2019-03-06 23:00:01 +07:00
import aiofiles
2016-07-21 18:15:35 +02:00
import aiohttp
import zipfile
2016-07-21 20:17:36 +02:00
import tempfile
2016-07-21 18:15:35 +02:00
2018-08-25 18:10:40 +07:00
from datetime import datetime
2016-07-21 18:15:35 +02:00
2016-10-26 16:50:01 +02:00
import logging
log = logging . getLogger ( __name__ )
2019-03-06 23:00:01 +07:00
CHUNK_SIZE = 1024 * 8 # 8KB
2016-10-26 16:50:01 +02:00
2024-07-06 17:08:16 +02:00
async def export_project ( zstream , project , temporary_dir , include_images = False , include_snapshots = False , keep_compute_ids = False , allow_all_nodes = False , reset_mac_addresses = False ) :
2016-07-21 18:15:35 +02:00
"""
2018-04-28 16:01:43 +07:00
Export a project to a zip file .
2016-07-21 18:15:35 +02:00
2018-04-28 16:01:43 +07:00
The file will be read chunk by chunk when you iterate over the zip stream .
Some files like snapshots and packet captures are ignored .
2016-07-21 18:15:35 +02:00
2019-03-06 23:00:01 +07:00
: param zstream : ZipStream object
: param project : Project instance
2016-07-21 20:17:36 +02:00
: param temporary_dir : A temporary dir where to store intermediate data
2019-03-07 18:55:38 +07:00
: param include_images : save OS images to the zip file
: param include_snapshots : save snapshots to the zip file
2024-07-06 17:08:16 +02:00
: param keep_compute_ids : If false replace all compute IDs y local ( standard behavior for . gns3project to make it portable )
: param allow_all_nodes : Allow all nodes type to be included in the zip even if not portable
: param reset_mac_addresses : Reset MAC addresses for each node .
2016-07-21 18:15:35 +02:00
"""
2018-04-28 16:01:43 +07:00
# To avoid issue with data not saved we disallow the export of a running project
2016-07-21 18:15:35 +02:00
if project . is_running ( ) :
2018-04-28 16:01:43 +07:00
raise aiohttp . web . HTTPConflict ( text = " Project must be stopped in order to export it " )
2016-07-21 18:15:35 +02:00
2017-02-27 11:08:58 +01:00
# Make sure we save the project
project . dump ( )
2016-11-25 17:18:23 +01:00
if not os . path . exists ( project . _path ) :
2018-04-28 16:01:43 +07:00
raise aiohttp . web . HTTPNotFound ( text = " Project could not be found at ' {} ' " . format ( project . _path ) )
2016-11-25 17:18:23 +01:00
2016-07-21 18:15:35 +02:00
# First we process the .gns3 in order to be sure we don't have an error
for file in os . listdir ( project . _path ) :
if file . endswith ( " .gns3 " ) :
2024-07-06 17:08:16 +02:00
await _patch_project_file ( project , os . path . join ( project . _path , file ) , zstream , include_images , keep_compute_ids , allow_all_nodes , temporary_dir , reset_mac_addresses )
2016-07-21 18:15:35 +02:00
2018-04-28 16:01:43 +07:00
# Export the local files
2019-02-19 12:43:44 +07:00
for root , dirs , files in os . walk ( project . _path , topdown = True , followlinks = False ) :
2021-03-24 13:16:00 +10:30
try :
files = [ f for f in files if _is_exportable ( os . path . join ( root , f ) , include_snapshots ) ]
for file in files :
path = os . path . join ( root , file )
2024-10-19 15:49:23 +10:00
if not os . path . islink ( path ) :
try :
# check if we can export the file
open ( path ) . close ( )
except OSError as e :
msg = " Could not export file {} : {} " . format ( path , e )
log . warning ( msg )
project . emit_notification ( " log.warning " , { " message " : msg } )
continue
2021-03-24 13:16:00 +10:30
# ignore the .gns3 file
if file . endswith ( " .gns3 " ) :
continue
_patch_mtime ( path )
zstream . write ( path , os . path . relpath ( path , project . _path ) )
2024-02-24 19:02:16 +08:00
# save empty directories
for directory in dirs :
path = os . path . join ( root , directory )
if not os . listdir ( path ) :
zstream . write ( path , os . path . relpath ( path , project . _path ) )
2021-03-24 13:16:00 +10:30
except FileNotFoundError as e :
log . warning ( " Cannot export local file: {} " . format ( e ) )
continue
2016-07-21 20:17:36 +02:00
2018-04-28 16:01:43 +07:00
# Export files from remote computes
2016-07-21 20:17:36 +02:00
for compute in project . computes :
2016-07-22 11:43:14 +02:00
if compute . id != " local " :
2018-10-15 17:05:49 +07:00
compute_files = await compute . list_files ( project )
2016-07-21 20:17:36 +02:00
for compute_file in compute_files :
2019-03-07 18:55:38 +07:00
if _is_exportable ( compute_file [ " path " ] , include_snapshots ) :
2019-03-06 23:00:01 +07:00
log . debug ( " Downloading file ' {} ' from compute ' {} ' " . format ( compute_file [ " path " ] , compute . id ) )
2018-10-15 17:05:49 +07:00
response = await compute . download_file ( project , compute_file [ " path " ] )
2021-03-24 13:16:00 +10:30
if response . status != 200 :
log . warning ( " Cannot export file from compute ' {} ' . Compute returned status code {} . " . format ( compute . id , response . status ) )
continue
2019-03-06 23:00:01 +07:00
( fd , temp_path ) = tempfile . mkstemp ( dir = temporary_dir )
async with aiofiles . open ( fd , ' wb ' ) as f :
while True :
try :
data = await response . content . read ( CHUNK_SIZE )
except asyncio . TimeoutError :
raise aiohttp . web . HTTPRequestTimeout ( text = " Timeout when downloading file ' {} ' from remote compute {} : {} " . format ( compute_file [ " path " ] , compute . host , compute . port ) )
if not data :
break
await f . write ( data )
2016-09-19 16:51:15 +02:00
response . close ( )
2018-08-25 18:10:40 +07:00
_patch_mtime ( temp_path )
2019-02-26 15:55:07 +07:00
zstream . write ( temp_path , arcname = compute_file [ " path " ] )
2017-11-13 22:12:39 +01:00
2016-07-21 18:15:35 +02:00
2018-08-25 18:10:40 +07:00
def _patch_mtime ( path ) :
"""
Patch the file mtime because ZIP does not support timestamps before 1980
: param path : file path
"""
if sys . platform . startswith ( " win " ) :
# only UNIX type platforms
return
2024-10-19 15:49:23 +10:00
st = os . stat ( path , follow_symlinks = False )
2018-08-25 18:10:40 +07:00
file_date = datetime . fromtimestamp ( st . st_mtime )
if file_date . year < 1980 :
new_mtime = file_date . replace ( year = 1980 ) . timestamp ( )
os . utime ( path , ( st . st_atime , new_mtime ) )
2019-02-19 12:43:44 +07:00
2019-03-07 18:55:38 +07:00
def _is_exportable ( path , include_snapshots = False ) :
2016-07-21 20:17:36 +02:00
"""
: returns : True if file should not be included in the final archive
"""
2016-07-26 10:32:43 +02:00
2019-03-07 18:55:38 +07:00
# do not export snapshots by default
if include_snapshots is False and path . endswith ( " snapshots " ) :
2018-04-28 16:01:43 +07:00
return False
2016-07-26 10:32:43 +02:00
2018-04-28 16:01:43 +07:00
# do not export directories of snapshots
2019-03-07 18:55:38 +07:00
if include_snapshots is False and " {sep} snapshots {sep} " . format ( sep = os . path . sep ) in path :
2018-04-28 16:01:43 +07:00
return False
2018-02-28 16:33:20 +01:00
2016-07-21 20:17:36 +02:00
try :
2018-04-28 16:01:43 +07:00
# do not export captures and other temporary directory
s = os . path . normpath ( path ) . split ( os . path . sep )
2016-07-21 20:17:36 +02:00
i = s . index ( " project-files " )
2019-03-07 18:55:38 +07:00
if include_snapshots is False and s [ i + 1 ] == " snapshots " :
return False
if s [ i + 1 ] in ( " tmp " , " captures " ) :
2018-04-28 16:01:43 +07:00
return False
2016-07-21 20:17:36 +02:00
except ( ValueError , IndexError ) :
pass
2018-04-28 16:01:43 +07:00
# do not export log files and OS noise
filename = os . path . basename ( path )
if filename . endswith ( ' _log.txt ' ) or filename . endswith ( ' .log ' ) or filename == ' .DS_Store ' :
return False
return True
2016-07-21 20:17:36 +02:00
2024-07-06 17:08:16 +02:00
async def _patch_project_file ( project , path , zstream , include_images , keep_compute_ids , allow_all_nodes , temporary_dir , reset_mac_addresses ) :
2016-07-21 18:15:35 +02:00
"""
2018-04-28 16:01:43 +07:00
Patch a project file ( . gns3 ) to export a project .
The . gns3 file is renamed to project . gns3
2016-07-21 18:15:35 +02:00
2018-04-28 16:01:43 +07:00
: param path : path of the . gns3 file
2016-07-21 18:15:35 +02:00
"""
2018-04-28 16:01:43 +07:00
# image files that we need to include in the exported archive
2017-11-13 22:12:39 +01:00
images = [ ]
2016-07-21 18:15:35 +02:00
2018-04-28 16:01:43 +07:00
try :
with open ( path ) as f :
topology = json . load ( f )
except ( OSError , ValueError ) as e :
raise aiohttp . web . HTTPConflict ( text = " Project file ' {} ' cannot be read: {} " . format ( path , e ) )
2016-07-21 18:15:35 +02:00
2016-07-22 13:39:57 +02:00
if " topology " in topology :
2016-07-25 14:47:37 +02:00
if " nodes " in topology [ " topology " ] :
for node in topology [ " topology " ] [ " nodes " ] :
2017-11-13 22:12:39 +01:00
compute_id = node . get ( ' compute_id ' , ' local ' )
2017-02-14 16:41:31 +01:00
if node [ " node_type " ] == " virtualbox " and node . get ( " properties " , { } ) . get ( " linked_clone " ) :
2018-04-28 16:01:43 +07:00
raise aiohttp . web . HTTPConflict ( text = " Projects with a linked {} clone node cannot not be exported. Please use Qemu instead. " . format ( node [ " node_type " ] ) )
2019-02-23 11:07:01 +07:00
if not allow_all_nodes and node [ " node_type " ] in [ " virtualbox " , " vmware " ] :
2018-04-28 16:01:43 +07:00
raise aiohttp . web . HTTPConflict ( text = " Projects with a {} node cannot be exported " . format ( node [ " node_type " ] ) )
2016-07-25 14:47:37 +02:00
2024-07-06 17:08:16 +02:00
if not keep_compute_ids :
2016-07-25 14:47:37 +02:00
node [ " compute_id " ] = " local " # To make project portable all node by default run on local
2024-09-26 18:41:23 +07:00
if " properties " in node :
2016-07-25 14:47:37 +02:00
for prop , value in node [ " properties " ] . items ( ) :
2017-11-13 22:12:39 +01:00
2019-02-20 16:38:43 +07:00
# reset the MAC address
if reset_mac_addresses and prop in ( " mac_addr " , " mac_address " ) :
node [ " properties " ] [ prop ] = None
2024-09-27 20:05:06 +07:00
if node [ " node_type " ] == " docker " :
continue
2018-03-12 13:38:50 +07:00
if node [ " node_type " ] == " iou " :
if not prop == " path " :
continue
elif not prop . endswith ( " image " ) :
continue
2017-11-13 22:12:39 +01:00
if value is None or value . strip ( ) == ' ' :
continue
2024-07-06 17:08:16 +02:00
if not keep_compute_ids : # If we keep the original compute we can keep the image path
2017-11-13 22:12:39 +01:00
node [ " properties " ] [ prop ] = os . path . basename ( value )
if include_images is True :
images . append ( {
' compute_id ' : compute_id ,
' image ' : value ,
' image_type ' : node [ ' node_type ' ]
} )
2016-07-25 14:47:37 +02:00
2024-07-06 17:08:16 +02:00
if not keep_compute_ids :
2017-11-13 22:12:39 +01:00
topology [ " topology " ] [ " computes " ] = [ ] # Strip compute information because could contain secret info like password
local_images = set ( [ i [ ' image ' ] for i in images if i [ ' compute_id ' ] == ' local ' ] )
for image in local_images :
2018-04-28 16:01:43 +07:00
_export_local_image ( image , zstream )
2017-11-13 22:12:39 +01:00
remote_images = set ( [
( i [ ' compute_id ' ] , i [ ' image_type ' ] , i [ ' image ' ] )
for i in images if i [ ' compute_id ' ] != ' local ' ] )
2016-07-22 13:39:57 +02:00
2017-11-13 22:12:39 +01:00
for compute_id , image_type , image in remote_images :
2018-10-15 17:05:49 +07:00
await _export_remote_images ( project , compute_id , image_type , image , zstream , temporary_dir )
2016-07-21 18:15:35 +02:00
2018-04-28 16:38:52 +07:00
zstream . writestr ( " project.gns3 " , json . dumps ( topology ) . encode ( ) )
2017-11-13 22:12:39 +01:00
return images
2016-07-21 18:15:35 +02:00
2019-02-26 15:55:07 +07:00
2018-04-28 16:01:43 +07:00
def _export_local_image ( image , zstream ) :
2016-07-21 18:15:35 +02:00
"""
2018-04-28 16:01:43 +07:00
Exports a local image to the zip file .
2016-07-21 18:15:35 +02:00
2018-04-28 16:01:43 +07:00
: param image : image path
: param zstream : Zipfile instance for the export
2016-07-21 18:15:35 +02:00
"""
2018-04-28 16:01:43 +07:00
from . . compute import MODULES
2016-07-21 18:15:35 +02:00
for module in MODULES :
try :
2018-04-28 16:01:43 +07:00
images_directory = module . instance ( ) . get_images_directory ( )
2016-07-21 18:15:35 +02:00
except NotImplementedError :
# Some modules don't have images
continue
2018-04-28 16:01:43 +07:00
directory = os . path . split ( images_directory ) [ - 1 : ] [ 0 ]
2016-07-21 18:15:35 +02:00
if os . path . exists ( image ) :
path = image
else :
2018-04-28 16:01:43 +07:00
path = os . path . join ( images_directory , image )
2016-07-21 18:15:35 +02:00
if os . path . exists ( path ) :
arcname = os . path . join ( " images " , directory , os . path . basename ( image ) )
2018-08-25 18:10:40 +07:00
_patch_mtime ( path )
2018-04-28 16:01:43 +07:00
zstream . write ( path , arcname )
2017-05-11 17:59:57 +02:00
return
2017-11-13 22:12:39 +01:00
2018-10-15 17:05:49 +07:00
async def _export_remote_images ( project , compute_id , image_type , image , project_zipfile , temporary_dir ) :
2017-11-13 22:12:39 +01:00
"""
2018-04-28 16:01:43 +07:00
Export specific image from remote compute .
2017-11-13 22:12:39 +01:00
"""
2019-03-06 23:00:01 +07:00
log . debug ( " Downloading image ' {} ' from compute ' {} ' " . format ( image , compute_id ) )
2017-11-13 22:12:39 +01:00
try :
compute = [ compute for compute in project . computes if compute . id == compute_id ] [ 0 ]
except IndexError :
2018-04-28 16:01:43 +07:00
raise aiohttp . web . HTTPConflict ( text = " Cannot export image from ' {} ' compute. Compute doesn ' t exist. " . format ( compute_id ) )
2017-11-13 22:12:39 +01:00
2018-10-15 17:05:49 +07:00
response = await compute . download_image ( image_type , image )
2017-11-13 22:12:39 +01:00
if response . status != 200 :
2019-03-06 23:00:01 +07:00
raise aiohttp . web . HTTPConflict ( text = " Cannot export image from compute ' {} ' . Compute returned status code {} . " . format ( compute_id , response . status ) )
2017-11-13 22:12:39 +01:00
2019-03-06 23:00:01 +07:00
( fd , temp_path ) = tempfile . mkstemp ( dir = temporary_dir )
async with aiofiles . open ( fd , ' wb ' ) as f :
while True :
try :
data = await response . content . read ( CHUNK_SIZE )
except asyncio . TimeoutError :
raise aiohttp . web . HTTPRequestTimeout ( text = " Timeout when downloading image ' {} ' from remote compute {} : {} " . format ( image , compute . host , compute . port ) )
if not data :
break
await f . write ( data )
2017-11-13 22:12:39 +01:00
response . close ( )
arcname = os . path . join ( " images " , image_type , image )
project_zipfile . write ( temp_path , arcname = arcname , compress_type = zipfile . ZIP_DEFLATED )