#!/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 .controller_error import ControllerNotFoundError 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): raise ControllerNotFoundError("Could not find symbol theme '{}'".format(theme)) self._current_theme = theme def default_symbols(self): return BUILTIN_SYMBOL_THEMES def get_default_symbol(self, symbol, symbol_theme): theme = self._themes.get(symbol_theme, None) if not theme: raise ControllerNotFoundError("Could not find symbol theme '{}'".format(symbol_theme)) symbol_path = theme.get(symbol) if symbol_path not in self._symbols_path: log.warning("Default symbol {} was not found".format(symbol_path)) return None return symbol_path 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('\\', '/') theme = posixpath.dirname(symbol_file).replace('/', '-').capitalize() if not theme: continue symbol_id = ':/symbols/' + symbol_file symbols.append({'symbol_id': symbol_id, 'filename': filename, 'theme': theme, '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('\\', '/') theme = posixpath.dirname(symbol_file).replace('/', '-').capitalize() if not theme: theme = "Custom symbols" symbols.append({'symbol_id': symbol_file, 'filename': filename, 'builtin': False, 'theme': theme}) self._symbols_path[symbol_file] = os.path.join(root, filename) symbols.sort(key=lambda x: x["filename"]) return symbols def symbols_path(self): server_config = Config.instance().settings.Server directory = os.path.expanduser(server_config.symbols_path) 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): try: return self._symbols_path[symbol_id] except KeyError: try: self.list() return self._symbols_path[symbol_id] except (OSError, KeyError): # try to return a symbol with the same name from the classic theme symbol = self._symbols_path.get(":/symbols/classic/{}".format(os.path.basename(symbol_id))) if symbol: return symbol else: # return the default computer symbol log.warning("Could not retrieve symbol '{}', returning default symbol...".format(symbol_id)) return self._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