# Copyright 2021 VyOS maintainers and contributors # # This library is free software; you can redistribute it and/or # modify it under the terms of the GNU Lesser General Public # License as published by the Free Software Foundation; either # version 2.1 of the License, or (at your option) any later version. # # This library 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 # Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public # License along with this library. If not, see . import os import sys import tempfile from ftplib import FTP import urllib.parse import urllib.request from vyos.util import cmd from paramiko import SSHClient def upload_ftp(local_path, hostname, remote_path,\ username='anonymous', password='', port=21): with open(local_path, 'rb') as file: with FTP() as conn: conn.connect(hostname, port) conn.login(username, password) conn.storbinary(f'STOR {remote_path}', file) def download_ftp(local_path, hostname, remote_path,\ username='anonymous', password='', port=21): with open(local_path, 'wb') as file: with FTP() as conn: conn.connect(hostname, port) conn.login(username, password) conn.retrbinary(f'RETR {remote_path}', file.write) def upload_sftp(local_path, hostname, remote_path,\ username=None, password=None, port=22): with SSHClient() as ssh: ssh.load_system_host_keys() ssh.connect(hostname, port, username, password) with ssh.open_sftp() as sftp: sftp.put(local_path, remote_path) def download_sftp(local_path, hostname, remote_path,\ username, password=None, port=22): with SSHClient() as ssh: ssh.load_system_host_keys() ssh.connect(hostname, port, username, password) with ssh.open_sftp() as sftp: sftp.get(remote_path, local_path) def upload_tftp(local_path, hostname, remote_path, port=69): with open(local_path, 'rb') as file: cmd(f'curl -s -T - tftp://{hostname}:{port}/{remote_path}', stderr=None, input=file.read()) def download_tftp(local_path, hostname, remote_path, port=69): with open(local_path, 'wb') as file: file.write(cmd(f'curl -s tftp://{hostname}:{port}/{remote_path}', stderr=None)) def download_http(urlstring, local_path): with open(local_path, 'wb') as file: with urllib.request.urlopen(urlstring) as response: file.write(response.read()) def get_remote_config(urlstring: str) -> bytes: """Download remote (config) file and return the contents. Args: remote file URI: scp://[:]@/ sftp://[:]@/ http:/// https:/// ftp://[[:]@]/ tftp:/// """ url = urllib.parse.urlparse(urlstring) temp = tempfile.NamedTemporaryFile(delete=False).name try: if url.scheme == 'http' or url.scheme == 'https': download_http(urlstring, temp) elif url.scheme == 'ftp': username = url.username if url.username else 'anonymous' download_ftp(temp, url.hostname, url.path, username, url.password) elif url.scheme == 'sftp' or url.scheme == 'scp': # None means we don't want to use password authentication. # An empty string (what urlparse returns when a password doesn't # exist in the URL) means the password is an empty string. password = url.password if url.password else None download_sftp(temp, url.hostname, url.path, url.username, password) elif url.scheme == 'tftp': download_tftp(temp, url.path, url.hostname, url.path) else: sys.exit('Unsupported URL scheme') with open(temp, 'r') as file: config = file.read() return config finally: os.remove(temp)