#!/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 posixpath from .symbol_themes import BUILTIN_SYMBOL_THEMES from ..utils.get_resource import get_resource from ..utils.picture import get_size from ..config import Config import logging log = logging.getLogger(__name__) class Symbols: """ Manage GNS3 symbols """ def __init__(self): try: self.list() except OSError: # The error will be raised and forwarded later pass # Keep a cache of symbols size self._symbol_size_cache = {} self._current_theme = "Classic" self._themes = BUILTIN_SYMBOL_THEMES @property def theme(self): return self._current_theme @theme.setter def theme(self, theme): if not self._themes.get(theme): log.error("Could not find symbol theme '{}'".format(theme)) return self._current_theme = theme def list(self): self._symbols_path = {} symbols = [] if get_resource("symbols"): for root, _, files in os.walk(get_resource("symbols")): for filename in files: if filename.startswith('.'): continue symbol_file = posixpath.normpath(os.path.relpath(os.path.join(root, filename), get_resource("symbols"))).replace('\\', '/') symbol_id = ':/symbols/' + symbol_file symbols.append({'symbol_id': symbol_id, 'filename': symbol_file, 'builtin': True}) self._symbols_path[symbol_id] = os.path.join(root, filename) directory = self.symbols_path() if directory: for root, _, files in os.walk(directory): for filename in files: if filename.startswith('.'): continue symbol_file = posixpath.normpath(os.path.relpath(os.path.join(root, filename), directory)).replace('\\', '/') symbols.append({'symbol_id': symbol_file, 'filename': symbol_file, 'builtin': False,}) self._symbols_path[symbol_file] = os.path.join(root, filename) symbols.sort(key=lambda x: x["filename"]) return symbols def symbols_path(self): directory = os.path.expanduser(Config.instance().get_section_config("Server").get("symbols_path", "~/GNS3/symbols")) if directory: try: os.makedirs(directory, exist_ok=True) except OSError as e: log.error("Could not create symbol directory '{}': {}".format(directory, e)) return None return directory def get_path(self, symbol_id): symbol_filename = os.path.splitext(os.path.basename(symbol_id))[0] theme = self._themes.get(self._current_theme, {}) if not theme: log.error("Could not find symbol theme '{}'".format(self._current_theme)) try: return self._symbols_path[theme.get(symbol_filename, symbol_id)] except KeyError: # Symbol not found, let's refresh the cache try: self.list() return self._symbols_path[symbol_id] except (OSError, KeyError): log.warning("Could not retrieve symbol '{}'".format(symbol_id)) symbols_path = self._symbols_path return symbols_path.get(":/symbols/classic/{}".format(os.path.basename(symbol_id)), symbols_path[":/symbols/classic/computer.svg"]) def get_size(self, symbol_id): try: return self._symbol_size_cache[symbol_id] except KeyError: with open(self.get_path(symbol_id), "rb") as f: res = get_size(f.read()) self._symbol_size_cache[symbol_id] = res return res