#!/usr/bin/env python3
##
## -----------------------------------------------------------------
## This file is part of WAPT Software Deployment
## Copyright (C) 2012 - 2024 Tranquil IT https://www.tranquil.it
## All Rights Reserved.
##
## WAPT helps systems administrators to efficiently deploy
## setup, update and configure applications.
## ------------------------------------------------------------------
##
from typing import Dict, List, Tuple, Optional, Any, Union, Callable
from waptutils import import_setup, format_bytes, wget, wgets, merge_dict, remove_encoding_declaration
from waptutils import httpdatetime2isodate, datetime2isodate, CustomZipFile, FileChunks, jsondump, LogOutput, isodate2datetime
from waptpackage import REGEX_PACKAGE_CONDITION, WaptLocalRepo, WaptRemoteRepo, PackageEntry, PackageRequest, HostCapabilities, PackageKey, PackageVersion
from waptcrypto import SSLCABundle, SSLCertificate, SSLPrivateKey, SSLCRL, SSLVerifyException
from waptutils import BaseObjectClass, ensure_list, ensure_unicode, default_http_headers, get_time_delta
from waptpackage import EWaptException, EWaptNotAPackage, EWaptNotSigned, EWaptBadPackageAttribute,EWaptBadSetup
from waptcrypto import get_peer_cert_chain_from_server, get_cert_chain_as_pem
from waptpackage import EWaptNeedsNewerAgent, EWaptDiskSpace
from waptcrypto import EWaptMissingPrivateKey, EWaptMissingCertificate, get_cert_chain_from_pem
from waptutils import isrunning, killalltasks, killtree, run
from waptutils import get_requests_client_cert_session, get_main_ip, update_ini_from_json_config, get_files_timestamp_sha256, hexdigest_for_data, sha256_for_data
from waptpackage import EWaptDownloadError, EWaptMissingPackageHook
from waptpackage import EWaptUnavailablePackage, EWaptConflictingPackage
import setuphelpers
from itsdangerous import URLSafeTimedSerializer
from waptpackage import make_valid_package_name
from waptutils import is_pem_key_encrypted
from waptutils import _disable_file_system_redirection
from waptutils import is_unsafe_filename
from waptutils import Timeit, config_overview
from waptutils import get_verify_cert
from waptutils import __version__
from waptutils import __file__ as waptutils__file__
import os
import re
import logging
import datetime
import time
import sys
import tempfile
import hashlib
import glob
import codecs
import base64
import zlib
import sqlite3
import json
import ujson
import io
import requests
import pickle
try:
# pylint: disable=no-member
# no error
import requests.packages.urllib3
from requests.packages.urllib3.exceptions import InsecureRequestWarning
requests.packages.urllib3.disable_warnings(InsecureRequestWarning)
except:
pass
import fnmatch
import ipaddress
import subprocess
import platform
import socket
import getpass
import psutil
import threading
import traceback
import uuid
import gc
import random
import string
from iniparse import RawConfigParser
from optparse import OptionParser
from operator import itemgetter
from collections import OrderedDict
from collections import defaultdict
import shutil
import urllib.parse
import zipfile
import arpy
import rpmfile
import tarfile
import imp
logger = logging.getLogger('waptcore')
tasks_logger = logging.getLogger('wapttasks')
bad_uuid= ['90218F6F-8B87-8442-8AAD-4A676F39B1F3','FFFFFFFF-FFFF-FFFF-FFFF-FFFFFFFFFFFF','00000000-0000-0000-0000-000000000000','Not Settable','03000200-0400-0500-0006-000700080009']
# conditionnal imports for windows or linux
if sys.platform == 'win32':
import win32api
import win32security
import win32net
import pywintypes
import pythoncom
from ntsecuritycon import DOMAIN_GROUP_RID_ADMINS
import winreg
from winreg import HKEY_LOCAL_MACHINE, EnumKey
old_argv = sys.argv
try:
import waptlicences
logger.debug('waptlicences module location: %s' % waptlicences.__file__)
except Exception as e:
logger.critical('Unable to load waptlicences module: %s' % e)
waptlicences = None
try:
import pyldap
except Exception as e:
logger.critical('Unable to load pyldap module: %s' % e)
pyldap = None
assert sys.argv == old_argv
if sys.platform == 'win32':
import pywaptwua
try:
import requests_kerberos
has_kerberos = True
except:
has_kerberos = False
if not sys.platform == 'win32':
import kerberos
else:
import winkerberos as kerberos
[docs]class EWaptBadServerAuthentication(EWaptException):
pass
[docs]def is_system_user():
return setuphelpers.get_current_user().lower() == 'system'
[docs]def tryurl(url, proxies=None, timeout=5.0, auth=None, verify_cert=False, cert=None):
# try to get header for the supplied URL, returns None if no answer within the specified timeout
# else return time to get he answer.
with get_requests_client_cert_session(url=url, cert=cert, verify=verify_cert, proxies=proxies) as session:
try:
logger.debug(' trying %s' % url)
starttime = time.time()
headers = session.head(url=url,
timeout=timeout,
auth=auth,
allow_redirects=True)
if headers.ok:
logger.debug(' OK')
return time.time() - starttime
else:
headers.raise_for_status()
except Exception as e:
logger.debug(' Not available : %s' % e)
return None
[docs]class EWaptCancelled(Exception):
pass
[docs]class WaptBaseDB(BaseObjectClass):
curr_db_version = None
def __init__(self, dbpath):
self._dbpath = ''
self._db= None
self.transaction_depth = 0
self.dbpath = dbpath
self.threadid = None
self.lock = threading.Lock()
@property
def dbpath(self):
return self._dbpath
@dbpath.setter
def dbpath(self, value):
if not self._dbpath or (self._dbpath and self._dbpath != value):
self._dbpath = value
self._db= None
[docs] def begin(self):
# recreate a connection if not in same thread (reuse of object...)
if self.transaction_depth == 0:
logger.debug('DB Start transaction')
self.execute('begin')
self.transaction_depth += 1
[docs] def commit(self):
if self.transaction_depth > 0:
self.transaction_depth -= 1
else:
msg = 'Unexpected commit of an already committed transaction...'
logger.critical(msg)
if logger.level == logging.DEBUG:
raise Exception(msg)
if self.transaction_depth == 0:
logger.debug('DB commit')
try:
self.execute('commit')
except:
self.execute('rollback')
raise
[docs] def rollback(self):
if self.transaction_depth > 0:
self.transaction_depth -= 1
if self.transaction_depth == 0:
logger.debug('DB rollback')
self.execute('rollback')
@property
def db(self) -> sqlite3.Connection:
if self._db is None: # or (self.threadid is not None and self.threadid != threading.get_ident()):
with self.lock:
"""
if self.threadid is not None and self.threadid != threading.get_ident():
logger.warning('Reset wapt.db (was created in thread %s but called from thread %s)' % (self.threadid, threading.get_ident()))
self._db = None
self.threadid = None
"""
if not self.dbpath:
raise EWaptException('dbpath not set. Unable to access core waptdb.sqlite database.')
logger.debug('Thread %s is connecting to wapt db' % threading.get_ident())
self.threadid = threading.get_ident()
if not self.dbpath == ':memory:' and not os.path.isfile(self.dbpath):
dirname = os.path.dirname(self.dbpath)
if os.path.isdir(dirname) == False:
os.makedirs(dirname)
os.path.dirname(self.dbpath)
self._db = sqlite3.connect(self.dbpath, detect_types=sqlite3.PARSE_DECLTYPES | sqlite3.PARSE_COLNAMES, check_same_thread=False)
self._db.isolation_level = None
self.transaction_depth = 0
self.create_db_structure()
self.init_db_data()
elif self.dbpath == ':memory:':
self._db = sqlite3.connect(self.dbpath, detect_types=sqlite3.PARSE_DECLTYPES | sqlite3.PARSE_COLNAMES, check_same_thread=False)
self._db.isolation_level = None
self.transaction_depth = 0
self.create_db_structure()
self.init_db_data()
else:
self._db = sqlite3.connect(self.dbpath, detect_types=sqlite3.PARSE_DECLTYPES | sqlite3.PARSE_COLNAMES, check_same_thread=False)
self._db.isolation_level = None
self.transaction_depth = 0
if self.curr_db_version != self.db_version:
self.upgradedb()
self.init_db_data()
return self._db
def __enter__(self):
self.start_timestamp = time.time()
self.begin()
#logger.debug(u'DB enter %i' % self.transaction_depth)
return self
def __exit__(self, type, value, tb):
if time.time()-self.start_timestamp > 1.0:
logger.debug('Transaction took too much time : %s' % (time.time()-self.start_timestamp,))
if not value:
#logger.debug(u'DB exit %i' % self.transaction_depth)
self.commit()
else:
self.rollback()
logger.debug('Error at DB exit %s, rollbacking\n%s' % (value, ensure_unicode(traceback.format_tb(tb))))
@property
def db_version(self):
val = self.execute('select value from wapt_params where name="db_version"').fetchone()
if val:
return val[0]
else:
return ''
@db_version.setter
def db_version(self, value):
with self:
self.execute('insert or replace into wapt_params(name,value,create_date) values (?,?,?)', ('db_version', value, datetime2isodate()))
@db_version.deleter
def db_version(self):
with self:
self.execute("delete from wapt_params where name = 'db_version'")
[docs] def create_db_structure(self):
return self.db_version
[docs] def set_param(self, name, value, ptype=None):
"""Store permanently a (name/value) pair in database, replace existing one"""
with self:
if not value is None:
if ptype is None:
if isinstance(value, str):
ptype = 'str'
if isinstance(value, bytes):
ptype = 'bytes'
# bool before int !
elif isinstance(value, bool):
ptype = 'bool'
elif isinstance(value, int):
ptype = 'int'
elif isinstance(value, float):
ptype = 'float'
elif isinstance(value, datetime.datetime):
ptype = 'datetime'
else:
ptype = 'json'
if ptype in ('int', 'float'):
value = str(value)
elif ptype in ('json', 'bool'):
value = jsondump(value)
elif ptype == 'datetime':
value = datetime2isodate(value)
elif ptype == 'bytes':
value = sqlite3.Binary(value)
self.execute('insert or replace into wapt_params(name,value,create_date,ptype) values (?,?,?,?)', (name, value, datetime2isodate(), ptype))
[docs] def has_param(self, name: str) -> bool:
"""Check if param exists"""
q = self.execute('select id from wapt_params where name=? limit 1', (name,)).fetchone()
if q:
return True
else:
return False
[docs] def get_param(self, name, default=None, ptype=None):
"""Retrieve the value associated with name from database"""
q = self.execute('select value,ptype from wapt_params where name=? order by create_date desc limit 1', (name,)).fetchone()
if q:
(value, sptype) = q
if ptype is None:
ptype = sptype
if not value is None:
if ptype == 'int':
value = int(value)
elif ptype == 'float':
value = float(value)
elif ptype in ('json', 'bool'):
value = ujson.loads(value)
elif ptype == 'datetime':
value = isodate2datetime(value)
return value
else:
return default
[docs] def delete_param(self, name):
with self:
row = self.execute('select value from wapt_params where name=? limit 1', (name,)).fetchone()
if row:
self.execute('delete from wapt_params where name=?', (name,))
[docs] def query(self, query, args=(), one=False, as_dict=True):
"""
execute la requete query sur la db et renvoie un tableau de dictionnaires
"""
cur = self.execute(query, args)
try:
if as_dict:
rv = [dict((cur.description[idx][0], value)
for idx, value in enumerate(row)) for row in cur.fetchall()]
else:
rv = cur.fetchall()
finally:
cur.close()
del cur
return (rv[0] if rv else None) if one else rv
[docs] def execute(self, query, args=()):
"""
Wrapper around sqlite.execute.
"""
nb_tries = 0
nb_max_tries = 3
seconds_before_retry = 2
while nb_tries < nb_max_tries:
try:
return self.db.execute(query, args)
except sqlite3.OperationalError as e:
if 'database is locked' in repr(e):
nb_tries += 1
if nb_tries >= nb_max_tries:
raise Exception("The database is locked. There is probably another WAPT process currently running (WaptAgent, WaptSelfService, wapt-get, waptpython.exe). Please check running processes. Error: %s" % (repr(e),))
else:
time.sleep(seconds_before_retry)
else:
raise
[docs] def init_db_data(self):
pass
[docs] def upgradedb(self, force=False):
"""Update local database structure to current version if rules are described in db_upgrades
Args:
force (bool): force upgrade even if structure version is greater than requested.
Returns:
tuple: (old_structure_version,new_structure_version)
"""
with self:
try:
backupfn = ''
# use cached value to avoid infinite loop
old_structure_version = self.db_version
if old_structure_version >= self.curr_db_version and not force:
logger.warning('upgrade db aborted : current structure version %s is newer or equal to requested structure version %s' % (old_structure_version, self.curr_db_version))
return (old_structure_version, old_structure_version)
logger.info('Upgrade database schema')
if self.dbpath != ':memory:':
# we will backup old data in a file so that we can rollback
backupfn = tempfile.mktemp('.sqlite')
logger.debug(' copy old data to %s' % backupfn)
shutil.copy(self.dbpath, backupfn)
else:
backupfn = None
# we will backup old data in dictionaries to convert them to new structure
logger.debug(' backup data in memory')
old_datas = {}
tables = [c[0] for c in self.execute('SELECT name FROM sqlite_master WHERE type = "table" and name like "wapt_%"').fetchall()]
for tablename in tables:
old_datas[tablename] = self.query('select * from %s' % tablename)
logger.debug(' %s table : %i records' % (tablename, len(old_datas[tablename])))
logger.debug(' drop tables')
for tablename in tables:
self.execute('drop table if exists %s' % tablename)
# create new empty structure
logger.debug(' recreates new tables ')
new_structure_version = self.create_db_structure()
del(self.db_version)
# append old data in new tables
logger.debug(' fill with old data')
for tablename in tables:
if old_datas[tablename]:
logger.debug(' process table %s' % tablename)
allnewcolumns = [c[0] for c in self.execute('select * from %s limit 0' % tablename).description]
# take only old columns which match a new column in new structure
oldcolumns = [k for k in old_datas[tablename][0] if k in allnewcolumns]
insquery = "insert into %s (%s) values (%s)" % (tablename, ",".join(oldcolumns), ",".join("?" * len(oldcolumns)))
for rec in old_datas[tablename]:
logger.debug(' %s' % [rec[oldcolumns[i]] for i in range(0, len(oldcolumns))])
self.execute(insquery, [rec[oldcolumns[i]] for i in range(0, len(oldcolumns))])
# be sure to put back new version in table as db upgrade has put the old value in table
self.db_version = new_structure_version
return (old_structure_version, new_structure_version)
except Exception as e:
if backupfn:
logger.critical("UpgradeDB ERROR : %s, copy back backup database %s" % (e, backupfn))
shutil.copy(backupfn, self.dbpath)
raise
[docs]class WaptSessionDB(WaptBaseDB):
curr_db_version = '20201217'
def __init__(self, username=''):
super(WaptSessionDB, self).__init__(None)
if not username:
username = setuphelpers.get_current_user()
self.username = username
self.dbpath = os.path.join(setuphelpers.application_data(), 'wapt', 'waptsession.sqlite')
[docs] def create_db_structure(self):
"""Initialize current sqlite db with empty table and return structure version"""
assert(isinstance(self.db, sqlite3.Connection))
logger.debug('Initialize Wapt session database')
self.execute("""
create table if not exists wapt_sessionsetup (
id INTEGER PRIMARY KEY AUTOINCREMENT,
username varchar,
package_uuid varchar,
package varchar,
version varchar,
architecture varchar,
install_date varchar,
install_status varchar,
install_output TEXT,
process_id integer
)"""
)
self.execute("""
create index if not exists idx_sessionsetup_username on wapt_sessionsetup(username,package);""")
self.execute("""
create index if not exists idx_sessionsetup_package on wapt_sessionsetup(package);""")
self.execute("""
create index if not exists idx_sessionsetup_package_uuid on wapt_sessionsetup(package_uuid);""")
self.execute("""
create table if not exists wapt_params (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name varchar,
value text,
ptype varchar,
create_date varchar
) """)
self.execute("""
create unique index if not exists idx_params_name on wapt_params(name);
""")
self.db_version = self.curr_db_version
return self.curr_db_version
[docs] def add_start_install(self, package_entry):
"""Register the start of installation in local db
Returns:
int : rowid of the inserted record
"""
with self:
cur = self.execute("""delete from wapt_sessionsetup where package=?""", (package_entry.package,))
cur = self.execute("""\
insert into wapt_sessionsetup (
username,
package_uuid,
package,
version,
architecture,
install_date,
install_status,
install_output,
process_id
) values (?,?,?,?,?,?,?,?,?)
""", (
self.username,
package_entry.package_uuid,
package_entry.package,
package_entry.version,
package_entry.architecture,
datetime2isodate(),
'INIT',
'',
os.getpid()
))
return cur.lastrowid
[docs] def update_install_status(self, rowid, set_status=None, append_line=None):
"""Update status of package installation on localdb"""
with self:
if set_status in ('OK', 'WARNING', 'ERROR'):
pid = None
else:
pid = os.getpid()
cur = self.execute("""\
update wapt_sessionsetup
set install_status=coalesce(?,install_status),install_output = coalesce(install_output,'') || ?,process_id=?
where rowid = ?
""", (
set_status,
ensure_unicode(append_line)+'\n' if append_line is not None else '',
pid,
rowid,
)
)
return cur.lastrowid
[docs] def update_install_status_pid(self, pid, set_status='ERROR'):
"""Update status of package installation on localdb"""
with self:
cur = self.execute("""\
update wapt_sessionsetup
set install_status=coalesce(?,install_status) where process_id = ?
""", (
set_status,
pid,
)
)
return cur.lastrowid
[docs] def remove_install_status(self, package):
"""Remove status of package installation from localdb
>>> wapt = Wapt()
>>> wapt.forget_packages('tis-7zip')
???
"""
with self:
cur = self.execute("""delete from wapt_sessionsetup where package=?""", (package,))
return cur.rowcount
[docs] def remove_obsolete_install_status(self, installed_packages) -> int:
"""Remove local user status of packages no more installed"""
with self:
cur = self.execute("""delete from wapt_sessionsetup where package not in (%s)""" %
','.join('?' for i in installed_packages), installed_packages)
return cur.rowcount
[docs] def is_installed(self, package, version):
p = self.query('select * from wapt_sessionsetup where package=? and version=? and install_status="OK"', (package, version))
if p:
return p[0]
else:
return None
[docs]class WaptPublicDB(WaptBaseDB):
curr_db_version = '20240930'
[docs] def create_db_structure(self) -> str:
"""Initialize current sqlite db with empty table and return structure version"""
assert(isinstance(self.db, sqlite3.Connection))
logger.debug('Initialize Wapt Public database')
self.execute("""
create table if not exists wapt_publicstatus (
id INTEGER PRIMARY KEY AUTOINCREMENT,
package_uuid varchar,
package varchar,
version varchar,
architecture varchar,
removal_date varchar,
control TEXT,
setuppy TEXT,
install_date varchar,
install_by varchar,
uninstall_date varchar,
uninstall_by varchar,
created_on varchar default (strftime('%Y-%m-%dT%H:%M:%fZ', 'now')),
updated_on varchar default (strftime('%Y-%m-%dT%H:%M:%fZ', 'now'))
)
""")
self.execute("""
create index if not exists idx_publicstatus_package_uuid on wapt_publicstatus(package_uuid);
""")
self.execute("""
create index if not exists idx_publicstatus_name on wapt_publicstatus(package,version);
""")
self.execute("""CREATE TRIGGER IF NOT EXISTS wapt_publicstatus_updated
BEFORE UPDATE ON wapt_publicstatus
BEGIN
update wapt_publicstatus set updated_on=strftime('%Y-%m-%dT%H:%M:%fZ', 'now') where package_uuid=NEW.package_uuid;
END
""")
self.execute("""
create table if not exists wapt_params (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name varchar,
value text,
ptype varchar,
create_date varchar
) """)
self.execute("""
create unique index if not exists idx_params_name on wapt_params(name);
""")
self.db_version = self.curr_db_version
return self.curr_db_version
[docs]class WaptDB(WaptBaseDB):
"""Class to manage SQLite database with local installation status"""
curr_db_version = '20240930'
wapt_package_columns = OrderedDict({
'id': 'INTEGER PRIMARY KEY AUTOINCREMENT',
'package_uuid': 'varchar',
'package': 'varchar',
'categories': 'varchar',
'version' : 'varchar',
'architecture': 'varchar',
'section': 'varchar',
'priority': 'varchar',
'maintainer': 'varchar',
'description': 'varchar',
'filename': 'varchar',
'size': 'integer',
'md5sum': 'varchar',
'depends': 'varchar',
'conflicts': 'varchar',
'sources': 'varchar',
'repo_url': 'varchar',
'repo': 'varchar',
'signer': 'varchar',
'signer_fingerprint': 'varchar',
'signature': 'varchar',
'signature_date': 'varchar',
'signed_attributes': 'varchar',
'min_wapt_version': 'varchar',
'maturity': 'varchar',
'locale': 'varchar',
'installed_size': 'integer',
'target_os': 'varchar',
'max_os_version': 'varchar',
'min_os_version': 'varchar',
'impacted_process': 'varchar',
'audit_schedule': 'varchar',
'name': 'varchar',
'editor': 'varchar',
'keywords': 'varchar',
'licence': 'varchar',
'homepage': 'varchar',
'changelog': 'varchar',
'valid_from': 'varchar',
'valid_until': 'varchar',
'forced_install_on': 'varchar',
'icon_sha256sum': 'varchar',
})
[docs] def create_db_structure(self) -> str:
"""Initialize current sqlite db with empty table and return structure version"""
assert(isinstance(self.db, sqlite3.Connection))
logger.debug('Initialize Wapt database')
query_cols = [col_name + ' ' + col_type for col_name, col_type in self.wapt_package_columns.items()]
query_cols = ','.join(query_cols)
init_wapt_package_table_query = 'create table if not exists wapt_package ({})'.format(query_cols)
self.execute(init_wapt_package_table_query)
self.execute("""
create index if not exists idx_package_name on wapt_package(package);""")
self.execute("""
create index if not exists idx_package_uuid on wapt_package(package_uuid);""")
self.execute("""
create table if not exists wapt_localstatus (
id INTEGER PRIMARY KEY AUTOINCREMENT,
package_uuid varchar,
package varchar,
version varchar,
version_pinning varchar,
explicit_by varchar,
architecture varchar,
section varchar,
priority varchar,
maturity varchar,
locale varchar,
install_date varchar,
install_status varchar,
install_output TEXT,
install_params VARCHAR,
uninstall_key varchar,
control TEXT,
setuppy TEXT,
process_id integer,
depends varchar,
conflicts varchar,
last_audit_on varchar,
last_audit_status varchar,
last_audit_output TEXT,
next_audit_on varchar,
impacted_process varchar,
audit_schedule varchar,
status_revision integer
)
""")
# in a separate table :
# upgrade_action -> 'INSTALL, UPGRADE, REMOVE'
# related_package_uuid -> package which will replace
# upgrade_planned_on
# upgrade_deadline
# upgrade_allowed_schedules
# retry_count
# max_retry_count
self.execute("""
create index if not exists idx_localstatus_name on wapt_localstatus(package);
""")
self.execute("""
create index if not exists idx_localstatus_status on wapt_localstatus(install_status);
""")
self.execute("""
create index if not exists idx_localstatus_next_audit_on on wapt_localstatus(next_audit_on);
""")
self.execute("""
create index if not exists idx_localstatus_package_uuid on wapt_localstatus(package_uuid);
""")
self.execute("""
create table if not exists wapt_params (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name varchar,
value text,
ptype varchar,
create_date varchar
) """)
self.execute("""
create unique index if not exists idx_params_name on wapt_params(name);
""")
self.execute("""CREATE TRIGGER IF NOT EXISTS inc_rev_ins_status
AFTER INSERT ON wapt_params
WHEN NEW.name not in ('status_revision','last_update_server_hashes')
BEGIN
update wapt_params set value=cast(value as integer)+1
where name='status_revision';
END
""")
self.execute("""CREATE TRIGGER IF NOT EXISTS inc_rev_upd_status
AFTER UPDATE ON wapt_params
WHEN NEW.name <> 'status_revision'
BEGIN
update wapt_params set value=cast(value as integer)+1
where name='status_revision';
END
""")
self.execute("""CREATE TRIGGER IF NOT EXISTS inc_rev_del_status
AFTER DELETE ON wapt_params
WHEN OLD.name <> 'status_revision'
BEGIN
update wapt_params set value=cast(value as integer)+1
where name='status_revision';
END
""")
self.execute("""CREATE TRIGGER IF NOT EXISTS wapt_localstatus_ai
AFTER INSERT ON wapt_localstatus
BEGIN
update wapt_localstatus set status_revision=(select cast(value as integer) from wapt_params where name='status_revision') WHERE id = NEW.id;
END
""")
self.execute("""CREATE TRIGGER IF NOT EXISTS wapt_localstatus_au
AFTER UPDATE ON wapt_localstatus
BEGIN
update wapt_localstatus set status_revision=(select cast(value as integer) from wapt_params where name='status_revision') WHERE id = NEW.id;
END
""")
self.execute("""CREATE TRIGGER IF NOT EXISTS wapt_localstatus_au
AFTER DELETE ON wapt_localstatus
BEGIN
update wapt_localstatus set status_revision=(select cast(value as integer) from wapt_params where name='status_revision') WHERE id = NEW.id;
END
""")
# action : install, remove, check, session_setup, update, upgrade
# state : draft, planned, postponed, running, done, error, canceled
self.execute("""
CREATE TABLE if not exists wapt_task (
id integer NOT NULL PRIMARY KEY AUTOINCREMENT,
action varchar,
state varchar,
current_step varchar,
process_id integer,
start_date varchar,
finish_date varchar,
package_name varchar,
username varchar,
package_version_min varchar,
package_version_max varchar,
rundate_min varchar,
rundate_max varchar,
rundate_nexttry varchar,
runduration_max integer,
created_date varchar,
run_params VARCHAR,
run_output TEXT
);
""")
self.execute("""
create index if not exists idx_task_state on wapt_task(state);
""")
self.execute("""
create index if not exists idx_task_package_name on wapt_task(package_name);
""")
self.execute("""
create table if not exists wapt_sessionsetup (
id INTEGER PRIMARY KEY AUTOINCREMENT,
username varchar,
package varchar,
version varchar,
architecture varchar,
maturity varchar,
locale varchar,
install_date varchar,
install_status varchar,
install_output TEXT
)"""
)
self.execute("""
create index idx_sessionsetup_username on wapt_sessionsetup(username,package);""")
# store metrics
# they are uploaded to server
self.execute("""
create table if not exists wapt_audit_data (
id INTEGER PRIMARY KEY AUTOINCREMENT,
value_date varchar,
value_section varchar,
value_key varchar,
value_type varchar,
value text,
expiration_date varchar
) """)
self.execute("""
create unique index idx_wapt_audit_data_key on wapt_audit_data(value_section,value_key,value_date);
""")
self.execute("""
create index idx_wapt_audit_data_date on wapt_audit_data(value_date);
""")
self.execute("""
create index idx_wapt_audit_data_exp on wapt_audit_data(expiration_date);
""")
self.db_version = self.curr_db_version
return self.curr_db_version
[docs] def init_db_data(self):
if self.get_param('status_revision') is None:
self.set_param('status_revision',0)
def _get_insert_package_query(self):
if not hasattr(self,'_insert_package_query'):
pkg_col_names = [col_name for col_name in self.wapt_package_columns]
pkg_col_names.remove('id')
pkg_col_names_str = ','.join(pkg_col_names)
self._insert_package_query = 'insert into wapt_package ({})'.format(pkg_col_names_str)
fields_nbr = len(pkg_col_names)
self._insert_package_query += ' values ({})'.format(','.join(['?'] * fields_nbr))
return self._insert_package_query
[docs] def add_package_entry(self, package_entry, locale_code=None) -> int:
"""Add a package into the database
"""
with self:
# for backward compatibility with packages signed without package_uuid attribute
if not package_entry.package_uuid:
package_entry.package_uuid = package_entry.make_fallback_uuid()
#cur = self.execute("""delete from wapt_package where package=? and version=? and target_os=? and architecture=? and maturity=? and locale=?""",
# (package_entry.package, package_entry.version, package_entry.target_os,package_entry.architecture, package_entry.maturity, package_entry.locale))
cur = self.execute("""delete from wapt_package where package_uuid=?""",(package_entry.package_uuid, ))
cur = self.execute(self._get_insert_package_query(), package_entry.get_values_for_db(locale_code))
return cur.lastrowid
[docs] def add_start_install(self, package_entry: PackageEntry, params_dict={}, explicit_by=None) -> int:
"""Register the start of installation in local db
Args:
package_entry (PackageEntry): package
params_dict (dict) : dictionary of parameters provided on command line with --param or by the server
explicit_by (str) : username of initiator of the install.
if not None, install is not a dependencie but an explicit manual install
setuppy (str) : python source code used for install, uninstall or session_setup
code used for uninstall or session_setup must use only wapt self library as
package content is no longer available at this step.
Returns:
int : rowid of the inserted install status row
"""
with self:
if package_entry.package_uuid:
# keep old entry for reference until install is completed.
cur = self.execute("""update wapt_localstatus set install_status='UPGRADING' where package=? and package_uuid <> ?""", (package_entry.package, package_entry.package_uuid))
cur = self.execute("""delete from wapt_localstatus where package_uuid=?""", (package_entry.package_uuid,))
else:
cur = self.execute("""delete from wapt_localstatus where package_uuid=?""", (package_entry.package,))
cur = self.execute("""\
insert into wapt_localstatus (
package_uuid,
package,
version,
section,
priority,
architecture,
install_date,
install_status,
install_output,
install_params,
explicit_by,
process_id,
maturity,
locale,
depends,
conflicts,
impacted_process,
audit_schedule,
control
) values (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)
""", (
package_entry.package_uuid,
package_entry.package,
package_entry.version,
package_entry.section,
package_entry.priority,
package_entry.architecture,
datetime2isodate(),
'INIT',
'',
jsondump(params_dict),
explicit_by,
os.getpid(),
package_entry.maturity,
package_entry.locale,
package_entry.depends,
package_entry.conflicts,
package_entry.impacted_process,
package_entry.audit_schedule,
package_entry._control_str()[0],
))
return cur.lastrowid
[docs] def update_install_status(self, rowid, set_status=None, append_line=None, uninstall_key=None) -> int:
"""Update status of package installation on localdb"""
with self:
if set_status in ('OK', 'WARNING', 'ERROR'):
pid = None
else:
pid = os.getpid()
cur = self.execute("""\
update wapt_localstatus
set install_status=coalesce(?,install_status),
install_output = coalesce(install_output,'') || ?,
uninstall_key=coalesce(?,uninstall_key),
process_id=?
where rowid = ?
""", (
set_status,
ensure_unicode(append_line)+'\n' if append_line is not None else '',
uninstall_key,
pid,
rowid,
)
)
# removed previously installed package entry
install_rec = self.query('select package_uuid,package from wapt_localstatus where rowid = ?', (rowid,), one=True)
if install_rec and set_status in ('OK', 'WARNING'):
cur = self.execute("""delete from wapt_localstatus where package=? and rowid <> ?""", (install_rec['package'], rowid))
return cur.lastrowid
[docs] def update_audit_status(self, rowid, set_status=None, set_output=None, append_line=None, set_last_audit_on=None, set_next_audit_on=None) -> int:
"""Update status of package installation on localdb"""
with self:
if set_status in ('OK', 'WARNING', 'ERROR'):
pid = None
else:
pid = os.getpid()
# retrieve last status
#cur = self.execute("""select last_audit_status,last_audit_on,next_audit_on from wapt_localstatus where rowid = ?""",(rowid,))
#(last_audit_status,last_audit_on,next_audit_on) = cur.fetchone()
# if last_audit_on is None:
# last_audit_on = datetime2isodate()
#
# if set_status is None:
# set_status = last_audit_status
# if set_status is None:
# set_status = 'RUNNING'
cur = self.execute("""\
update wapt_localstatus set
last_audit_status=coalesce(?,last_audit_status,'RUNNING'),
last_audit_on=coalesce(?,last_audit_on),
last_audit_output = coalesce(?,last_audit_output,'') || ?,
process_id=?,next_audit_on=coalesce(?,next_audit_on)
where rowid = ?
""", (
set_status,
set_last_audit_on,
set_output,
append_line+'\n' if append_line is not None else '',
pid,
set_next_audit_on,
rowid
)
)
return cur.lastrowid
[docs] def update_install_status_pid(self, pid, set_status='ERROR') -> int:
"""Update status of package installation on localdb"""
with self:
cur = self.execute("""\
update wapt_localstatus
set install_status=coalesce(?,install_status) where process_id = ?
""", (
set_status,
pid,
)
)
return cur.lastrowid
[docs] def switch_to_explicit_mode(self, package, user_id) -> int:
"""Set package install mode to manual
so that package is not removed
when meta packages don't require it anymore
"""
with self:
cur = self.execute("""\
update wapt_localstatus
set explicit_by=? where package = ?
""", (
user_id,
package,
)
)
return cur.lastrowid
[docs] def remove_install_status(self, package=None, package_uuid=None):
"""Remove status of package installation from localdb"""
with self:
if package_uuid is not None:
cur = self.execute("""delete from wapt_localstatus where package_uuid=?""", (package_uuid,))
else:
cur = self.execute("""delete from wapt_localstatus where package=?""", (package,))
return cur.rowcount
[docs] def known_packages(self):
"""Return a dict of all known packages PackageKey(s) indexed by package_uuid
Returns:
dict {'package_uuid':PackageKey(package)}
"""
q = self.execute("""\
select distinct wapt_package.package_uuid,wapt_package.package,wapt_package.version,architecture,locale,maturity from wapt_package
""")
return {e[0]: PackageKey(e[0], e[1], Version(e[2]), *e[3:]) for e in q.fetchall()}
[docs] def packages_matching(self, package_cond):
"""Return an ordered list of available packages entries which match
the condition "packagename[([=<>]version)]?"
version ascending
Args:
package_cond (PackageRequest or str): filter packages and determine the ordering
Returns:
list of PakcageEntry
"""
if isinstance(package_cond, str):
package_cond = PackageRequest(request=package_cond)
if package_cond.package_uuid is not None:
q = self.query_package_entry("""\
select * from wapt_package where package_uuid = ?
""", (package_cond.package_uuid,))
elif package_cond.package is not None:
q = self.query_package_entry("""\
select * from wapt_package where package = ?
""", (package_cond.package,))
else:
q = self.query_package_entry("""\
select * from wapt_package
""")
result = [p for p in q if package_cond.is_matched_by(p)]
result.sort(key=package_cond.get_package_compare_key)
return result
[docs] def packages_search(self, searchwords=[], exclude_host_repo=True, section_filter=None, packages_filter=None):
"""Return a list of package entries matching the search words
Args:
searchwords (list): list of words which must be in package name or description
exclude_host (bool): don't take in account packages comming from a repo named 'wapt-host"
section_filter (list): list of packages sections to take in account
packages_filter (PackageRequest): additional filters (arch, locale, maturities etc...)
to take in account for filter and sort
Returns:
list of PackageEntry
"""
if not isinstance(searchwords, list) and not isinstance(searchwords, tuple):
searchwords = [searchwords]
if not searchwords:
words = []
search = ['1=1']
else:
words = ["%"+w.lower()+"%" for w in searchwords]
search = ["lower(description || package) like ?"] * len(words)
if exclude_host_repo:
search.append('repo <> "wapt-host"')
if section_filter:
section_filter = ensure_list(section_filter)
search.append('section in ( %s )' % ",".join(['"%s"' % x for x in section_filter]))
if isinstance(packages_filter, str):
packages_filter = PackageRequest(request=packages_filter)
result = self.query_package_entry("select * from wapt_package where %s" % " and ".join(search), words)
if packages_filter is not None:
result = [p for p in result if packages_filter.is_matched_by(packages_filter)]
result.sort(key=packages_filter.get_package_compare_key)
else:
result.sort()
return result
[docs] def installed_status(self, include_errors=False, include_setup=False, include_control=False, include_log=False):
"""Return a list of installed packages on this host (status 'OK' or 'UNKNWON')
Args:
include_errors (bool) : if False, only packages with status 'OK' and 'UNKNOWN' are returned
if True, all packages are installed.
include_setup (bool) : if True, setup.py files content is in the result rows
Returns:
dict[package_uuid]: install and audit infos
"""
sql = ["""\
select l.package_uuid,l.package,l.version,l.install_date,l.install_status,l.install_params,l.explicit_by,
l.last_audit_status,l.last_audit_on,%s%s%s
l.next_audit_on
from wapt_localstatus l
""" % (('l.setuppy,' if include_setup else ''),('l.control,' if include_control else ''), ('l.install_output,L.last_audit_output,' if include_log else ''))]
if not include_errors:
sql.append('where l.install_status in ("OK","UNKNOWN")')
q = self.query('\n'.join(sql))
return list(q)
[docs] def installed(self, include_errors=False, include_setup=True, include_control=False):
"""Return a list of installed packages on this host (status 'OK' or 'UNKNWON')
Args:
include_errors (bool) : if False, only packages with status 'OK' and 'UNKNOWN' are returned
if True, all packages are installed.
include_setup (bool) : if True, setup.py files content is in the result rows
Returns:
list: of installed PackageEntry
"""
sql = ["""\
select l.id,l.package,l.version,l.architecture,r.description,r.name,l.install_date,l.install_status,l.install_output,l.install_params,%s
l.uninstall_key,l.explicit_by,
coalesce(l.depends,r.depends) as depends,coalesce(l.conflicts,r.conflicts) as conflicts,coalesce(l.section,r.section) as section,
coalesce(l.priority,r.priority) as priority,
r.maintainer,r.name,r.description,r.sources,r.filename,r.size,
r.repo_url,r.md5sum,r.repo,r.signer,r.signature_date,r.signer_fingerprint,
l.maturity,l.locale,
l.last_audit_status,l.last_audit_on,l.last_audit_output,l.next_audit_on,%s
l.package_uuid
from wapt_localstatus l
left join wapt_package r on r.package_uuid=l.package_uuid
""" % (('l.setuppy,' if include_setup else ''),('l.control,' if include_control else ''))]
if not include_errors:
sql.append('where l.install_status in ("OK","UNKNOWN")')
q = self.query_package_entry('\n'.join(sql))
result = []
for p in q:
result.append(p)
return result
[docs] def installed_packages_inventory(self,since_status_revision=None):
"""Return a list of installed packages status suitable for inventory
List of list.
First line is the header (fields names)
Returns:
list (of list): of installed packages
"""
query = """\
select l.id,l.package,l.version,l.architecture,l.install_date,l.install_status,l.install_output,l.install_params,
l.uninstall_key,l.explicit_by,
coalesce(l.depends,r.depends) as depends,
coalesce(l.conflicts,r.conflicts) as conflicts,
coalesce(l.section,r.section) as section,
coalesce(l.priority,r.priority) as priority,
r.maintainer,r.name,r.description,r.sources,r.filename,r.size,
r.repo_url,r.md5sum,r.repo,r.signer,r.signature_date,r.signer_fingerprint,
l.maturity,l.locale,
l.package_uuid
from wapt_localstatus l
left join wapt_package r on r.package_uuid=l.package_uuid
"""
if since_status_revision is None:
cur = self.execute(query)
else:
cur = self.execute(query+' where l.status_revision>?',[since_status_revision])
result = []
# first line is columns headers
result.append([c[0] for c in cur.description])
for row in cur.fetchall():
result.append(row)
return result
[docs] def installed_packages_ids(self):
"""Return list of id of package install status
"""
cur = self.execute('select id from wapt_localstatus')
return [c[0] for c in cur.fetchall()]
[docs] def packages_audit_inventory(self,after_date=None):
"""Return a list of packages audit status suitable for inventory
List of list.
First line is the header (fields names)
Returns:
list (of list): of installed packages
"""
query = """\
select l.id,l.package_uuid,
l.last_audit_status,l.last_audit_on,l.last_audit_output,l.next_audit_on
from wapt_localstatus l
"""
if after_date is None:
cur = self.execute(query)
else:
cur = self.execute(query+' where l.last_audit_on > ?',(after_date,))
result = []
result.append([c[0] for c in cur.description])
for row in cur.fetchall():
result.append(list(row))
return result
[docs] def install_status(self, id):
"""Return the local install status for id
Args:
id: sql rowid
Returns:
dict : merge of package local install, audit and package attributes.
"""
sql = ["""\
select l.package,l.version,l.architecture,l.install_date,l.install_status,l.install_output,l.install_params,l.explicit_by,
l.depends,l.conflicts,l.uninstall_key,l.setuppy,l.control,
l.last_audit_status,l.last_audit_on,l.last_audit_output,l.next_audit_on,l.audit_schedule,l.package_uuid,
r.section,r.priority,r.maintainer,r.description,r.sources,r.filename,r.size,r.signer,r.signature_date,r.signer_fingerprint,
r.repo_url,r.md5sum,r.repo,l.maturity,l.locale
from wapt_localstatus l
left join wapt_package r on
r.package_uuid=l.package_uuid
where l.id = ?
"""]
q = self.query_package_entry('\n'.join(sql), args=[id])
if q:
return q[0]
else:
return None
[docs] def installed_search(self, searchwords=[], include_errors=False):
"""Return a list of installed package entries based on search keywords
Returns:
list of PackageEntry merge with localstatus attributes without setuppy
"""
if not isinstance(searchwords, list) and not isinstance(searchwords, tuple):
searchwords = [searchwords]
if not searchwords:
words = []
search = ['1=1']
else:
words = ["%"+w.lower()+"%" for w in searchwords]
search = ["lower(l.package || (case when r.description is NULL then '' else r.description end) ) like ?"] * len(words)
if not include_errors:
search.append('l.install_status in ("OK","UNKNOWN")')
q = self.query_package_entry("""\
select l.package,l.version,l.architecture,l.install_date,l.install_status,l.install_output,l.install_params,
l.uninstall_key,l.explicit_by,
coalesce(l.depends,r.depends) as depends,coalesce(l.conflicts,r.conflicts) as conflicts,coalesce(l.section,r.section) as section,coalesce(l.priority,r.priority) as priority,
l.last_audit_status,l.last_audit_on,l.last_audit_output,l.next_audit_on,l.audit_schedule,l.package_uuid,
r.maintainer,r.description,r.sources,r.filename,r.size,r.signer,r.signature_date,r.signer_fingerprint,
r.repo_url,r.md5sum,r.repo
from wapt_localstatus l
left join wapt_package r on r.package_uuid=l.package_uuid
where %s
""" % " and ".join(search), words)
return q
[docs] def installed_matching(self, package_cond:str, include_errors=False, include_setup=True, include_control=False) -> PackageEntry:
"""Return the properly installed (if include_errors=False) package match the package condition 'tis-package (>=version)'
Args:
package_cond (str): package requirement to lookup
include_errors
Returns:
PackageEntry merge with localstatus attributes WITH setuppy
"""
if isinstance(package_cond, str):
requ = package_cond
package_cond = PackageRequest(request=requ)
elif not isinstance(package_cond, PackageRequest):
raise Exception('installed_matching: package_cond must be either str ot PackageRequest')
if include_errors:
status = '"OK","UNKNOWN","ERROR"'
else:
status = '"OK","UNKNOWN"'
q = self.query_package_entry("""\
select l.rowid,l.package_uuid,
l.package,l.version,l.architecture,
coalesce(l.locale,r.locale) as locale,
coalesce(l.maturity,r.maturity) as maturity,
l.install_date,l.install_status,l.install_output,l.install_params,%s
l.uninstall_key,l.explicit_by,
l.last_audit_status,l.last_audit_on,l.last_audit_output,l.next_audit_on,
coalesce(l.depends,r.depends) as depends,
coalesce(l.conflicts,r.conflicts) as conflicts,
coalesce(l.section,r.section) as section,
coalesce(l.priority,r.priority) as priority,%s
r.maintainer,r.description,r.sources,r.filename,r.size,r.signer,r.signature_date,r.signer_fingerprint,
r.repo_url,r.md5sum,r.repo
from wapt_localstatus l
left join wapt_package r on r.package_uuid=l.package_uuid
where (l.package=? or l.package_uuid=?) and l.install_status in (%s)
order by l.install_date
""" % (('l.setuppy,' if include_setup else ''), ('l.control,' if include_control else ''), status), (package_cond.package,package_cond.package_uuid))
return q[0] if q and package_cond.is_matched_by(q[0]) else None
[docs] def upgradeable(self, include_errors=True, check_version_only=True,include_setup=False, include_control=False):
"""Return a dictionary of upgradable Package entries
Returns:
dict 'package': [candidates] order by mot adequate to more generic and
"""
result = {}
allinstalled = self.installed(include_errors=True,include_setup=include_setup,include_control=include_control)
for p in allinstalled:
available = self.query_package_entry("""select * from wapt_package where package=?""", (p.package,))
available.sort()
available.reverse()
if (available and (
((check_version_only and PackageVersion(available[0]['version']) > PackageVersion(p['version'])) or (not check_version_only and available[0] > p)) or # check version only or full attributes
(include_errors and p.install_status == 'ERROR'))
): # if current package is in error status, and we have a candidate, add it
result[p.package] = available
return result
[docs] def upgradeable_status(self):
"""Return a list of upgradable Package entries. (faster than upgradeable() )
Returns:
list of (installed PackageEntry, candidate PackageEntry)
"""
result = []
allinstalled = self.query("""select install_status,package_uuid,package,version,section,priority,maturity,locale from wapt_localstatus""")
for p in allinstalled:
availables = self.query("""select package_uuid,package,version,section,priority,maturity,locale,target_os from wapt_package where (? in ('RETRY','ERROR') or package_uuid<>?) and package=?""", (p['install_status'],p['package_uuid'],p['package'],))
upgr = None
for available in availables:
if available and (p['install_status'] in ('ERROR','RETRY') or PackageVersion(available['version']) > PackageVersion(p['version'])):
if upgr is None or PackageVersion(available['version']) > PackageVersion(upgr['version']):
upgr = available
if upgr:
result.append((PackageEntry(**p),PackageEntry(**upgr)))
return result
[docs] def audit_status(self):
"""Return WORST audit status among properly installed packages"""
errors = self.query("""select count(*) from wapt_localstatus where install_status="OK" and last_audit_status="ERROR" """, one=True, as_dict=False)[0]
if errors > 0:
return 'ERROR'
warnings = self.query("""select count(*) from wapt_localstatus where install_status="OK" and (last_audit_status is NULL or last_audit_status in ("WARNING","UNKNOWN")) """, one=True, as_dict=False)[0]
if warnings and warnings > 0:
return 'WARNING'
return 'OK'
[docs] def build_depends(self, packages, packages_filter=None):
"""Given a list of packages conditions (packagename (optionalcondition))
return a list of dependencies (packages conditions) to install
Args:
packages (list of str): list of packages requirements ( package_name(=version) )
Returns:
(list depends,list conflicts,list missing) : tuple of (all_depends,missing_depends)
TODO : choose available dependencies in order to reduce the number of new packages to install
>>> waptdb = WaptDB(':memory:')
>>> office = PackageEntry('office','0')
>>> firefox22 = PackageEntry('firefox','22')
>>> firefox22.depends = 'mymissing,flash'
>>> firefox24 = PackageEntry('firefox','24')
>>> thunderbird = PackageEntry('thunderbird','23')
>>> flash10 = PackageEntry('flash','10')
>>> flash12 = PackageEntry('flash','12')
>>> office.depends='firefox(<24),thunderbird,mymissing'
>>> firefox22.depends='flash(>=10)'
>>> firefox24.depends='flash(>=12)'
>>> waptdb.add_package_entry(office)
>>> waptdb.add_package_entry(firefox22)
>>> waptdb.add_package_entry(firefox24)
>>> waptdb.add_package_entry(flash10)
>>> waptdb.add_package_entry(flash12)
>>> waptdb.add_package_entry(thunderbird)
>>> waptdb.build_depends('office')
([u'flash(>=10)', u'firefox(<24)', u'thunderbird'], [u'mymissing'])
"""
if not isinstance(packages, list) and not isinstance(packages, tuple):
packages = [packages]
MAXDEPTH = 30
# roots : list of initial packages to avoid infinite loops
alldepends = []
allconflicts = []
missing = []
explored = []
def dodepends(packages, depth):
if depth > MAXDEPTH:
raise Exception('Max depth in build dependencies reached, aborting')
package_request = PackageRequest(request=None, copy_from=packages_filter)
# loop over all package names
for package in packages:
if not package in explored:
if isinstance(package, str):
package_request.request = package
entries = self.packages_matching(package_request)
else:
entries = self.packages_matching(package)
if not entries and package not in missing:
missing.append(package)
else:
# get depends of the most recent matching entry
# TODO : use another older if this can limit the number of packages to install !
depends = ensure_list(entries[-1].depends)
available_depends = []
for d in depends:
package_request.request = d
if self.packages_matching(package_request):
available_depends.append(d)
elif d not in missing:
missing.append(d)
newdepends = dodepends(available_depends, depth+1)
for d in newdepends:
if not d in alldepends:
alldepends.append(d)
for d in available_depends:
if not d in alldepends:
alldepends.append(d)
conflicts = ensure_list(entries[-1].conflicts)
for d in conflicts:
if not d in allconflicts:
allconflicts.append(d)
explored.append(package)
return alldepends
depth = 0
alldepends = dodepends(packages, depth)
return (alldepends, allconflicts, missing)
[docs] def query_package_entry(self, query, args=(), one=False, package_request=None):
"""Execute the query on the db try to map result on PackageEntry attributes
Fields which don't match attributes are added as attributes (and listed in _calc_attributes list)
Args:
query (str): sql query
args (list): parameters of the sql query
package_request (PackageRequest): keep only entries which match package_request
one(bool): if True, return the highest available version (if package_request is not None, use it to compare packages)
Result:
PackageEntry or list of PackageEntry
>>> waptdb = WaptDB(':memory:')
>>> waptdb.add_package_entry(PackageEntry('toto','0',repo='main'))
>>> waptdb.add_package_entry(PackageEntry('dummy','2',repo='main'))
>>> waptdb.add_package_entry(PackageEntry('dummy','1',repo='main'))
>>> waptdb.query_package_entry("select * from wapt_package where package=?",["dummy"])
[PackageEntry('dummy','2'), PackageEntry('dummy','1')]
>>> waptdb.query_package_entry("select * from wapt_package where package=?",["dummy"],one=True)
PackageEntry('dummy','2')
"""
result = []
cur = self.execute(query, args)
for row in cur.fetchall():
pe = PackageEntry()
rec_dict = dict((cur.description[idx][0], value) for idx, value in enumerate(row))
for k in rec_dict:
setattr(pe, k, rec_dict[k])
# add joined field to calculated attributes list
if not k in pe.all_attributes:
pe._calculated_attributes.append(k)
if package_request is None or package_request.is_matched_by(pe):
result.append(pe)
if one and result:
if not package_request:
result = sorted(result)[-1]
else:
result = sorted(result, key=package_request.get_package_compare_key)[-1]
return result
[docs] def purge_repo(self, repo_name):
"""remove references to repo repo_name
>>> waptdb = WaptDB('c:/wapt/db/waptdb.sqlite')
>>> waptdb.purge_repo('main')
"""
with self:
self.execute('delete from wapt_package where repo=?', (repo_name,))
[docs] def params(self, packagename):
"""Return install parameters associated with a package"""
with self:
cur = self.execute("""select install_params from wapt_localstatus where package=?""", (packagename,))
rows = cur.fetchall()
if rows:
return ujson.loads(rows[0][0])
[docs] def get_packages_from_uuids(self,uuids):
with self:
cur = self.query_package_entry("""select * from wapt_packages where package_uuid in (%s)""" % ','.join('?'*len(uuids)), uuids)
rows = cur.fetchall()
return list(rows)
[docs] def get_not_installed_from_uuids(self,uuids):
with self:
cur = self.query_package_entry("""select p.* from wapt_packages p
left join wapt_localstatus s on s.package_uuid=p.package_uuid
where p.package_uuid in (%s) and s.id is null""" % ','.join('?'*len(uuids)), uuids)
rows = cur.fetchall()
return list(rows)
[docs] def get_status_revision(self):
q = self.execute("""select cast(value as integer) from wapt_params where name='status_revision' limit 1""").fetchone()
if q:
return q[0]
else:
return None
[docs]class WaptServer(BaseObjectClass):
"""Manage connection to waptserver"""
def __init__(self, url=None, proxies={'http': None, 'https': None}, timeout=5.0, dnsdomain=None, name='waptserver'):
if url and url[-1] == '/':
url = url.rstrip('/')
self._server_url = url
self.name = name
self.proxies = proxies
self.timeout = timeout
self.use_kerberos = False
self.verify_cert = True
self.client_certificate = None
self.client_private_key = None
self.interactive_session = False
self.ask_user_password_hook = None
self.private_key_password_callback = None
self.capture_external_ip_callback = None
if dnsdomain:
self.dnsdomain = dnsdomain
else:
self.dnsdomain = setuphelpers.get_domain()
self.clear_session()
[docs] def auth(self, action=None, enable_password_callback=True):
"""Return an authenticator for the action.
This is basically a tuple (user,password) or a more sophisticated AuthBase descendent.
The motivation for action is to not request password from user if not needed.
Or enable kerberos for specific endpoint.
Args:
action (str): for which an authenticator is requested
Returns:
tuple or requests.auth.AuthBase descendant instance
"""
if self._server_url:
if action in ('add_host_kerberos', 'add_host'):
if not (sys.platform == 'win32'):
try:
# TODO found other method for TGS
setuphelpers.get_domain_info()
except:
pass
scheme = urllib.parse.urlparse(self._server_url).scheme
if scheme == 'https' and has_kerberos and self.use_kerberos:
# TODO : simple auth if kerberos is not available...
return requests_kerberos.HTTPKerberosAuth(mutual_authentication=requests_kerberos.DISABLED)
if enable_password_callback:
return self.ask_user_password(action)
else:
return None
else:
return None
[docs] def get_private_key_password(self, location, identity):
if self.private_key_password_callback is not None:
return self.private_key_password_callback(location, identity)
else:
return None
[docs] def clear_session(self):
self._session = None
self._session_url = None
self._session_client_certificate = None
self._session_use_ssl_auth = None
[docs] def get_requests_session(self, use_ssl_auth=True):
# don't use cached session if parameters have changed
if use_ssl_auth != self._session_use_ssl_auth or \
(self._session_use_ssl_auth and self._session_client_certificate != self.client_certificate):
self.clear_session()
if self._session is None:
url = self.server_url
if use_ssl_auth:
if self.client_private_key and is_pem_key_encrypted(self.client_private_key):
password = self.get_private_key_password(url, self.client_private_key)
else:
password = None
cert = (self.client_certificate, self.client_private_key, password)
else:
cert = None
self._session = get_requests_client_cert_session(url=url, cert=cert, verify=self.verify_cert, proxies=self.proxies)
self._session_use_ssl_auth = use_ssl_auth
self._session_url = url
self._session_client_certificate = self.client_certificate
self._session.headers = default_http_headers()
return self._session
[docs] def save_server_certificate(self, server_ssl_dir=None, overwrite=False):
"""Retrieve certificate of https server for further checks
Args:
server_ssl_dir (str): Directory where to save x509 certificate file
Returns:
str : full path to x509 certificate file.
"""
certs = get_peer_cert_chain_from_server(self.server_url)
if certs:
new_cert = certs[0]
url = urllib.parse.urlparse(self.server_url)
pem_fn = os.path.join(server_ssl_dir, new_cert.cn.replace('*','wildcard') +'.crt')
if not fnmatch.fnmatch(url.hostname, new_cert.cn) :
logger.warning('Warning, certificate CN %s sent by server does not match URL host %s' % (new_cert.cn, url.hostname))
if not os.path.isdir(server_ssl_dir):
os.makedirs(server_ssl_dir)
if os.path.isfile(pem_fn):
try:
# compare current and new cert
old_cert = SSLCertificate(pem_fn)
if old_cert.modulus != new_cert.modulus:
if not overwrite:
raise Exception('Can not save server certificate, a file with same name but from diffrent key already exists in %s' % pem_fn)
else:
logger.info('Overwriting old server certificate %s with new one %s' % (old_cert.fingerprint, new_cert.fingerprint))
return pem_fn
except Exception as e:
logger.critical('save_server_certificate : %s' % repr(e))
raise
# write full chain
with open(pem_fn, 'w',encoding = 'utf8') as f:
f.write(get_cert_chain_as_pem(certs))
logger.info('New certificate %s with fingerprint %s saved to %s' % (new_cert, new_cert.fingerprint, pem_fn))
return pem_fn
else:
return None
[docs] def reset_network(self):
"""called by wapt when network configuration has changed"""
self.clear_session()
@property
def server_url(self):
"""Return fixed url if any
>>> server = WaptServer(timeout=4)
>>> print server.server_url
https://wapt.tranquil.it
"""
return self._server_url
@server_url.setter
def server_url(self, value):
"""Wapt main repository URL
The URL is explicitly set (stored in private _server_url)
Returns:
str : URL of wapt server root
"""
# remove / at the end
if value:
value = value.rstrip('/')
if value != self._server_url:
self.clear_session()
self._server_url = value
[docs] def load_config(self, config, section='global'):
"""Load waptserver configuration from inifile
"""
if not section:
section = 'global'
if config.has_section(section):
self.name = section
if config.has_option(section, 'wapt_server'):
# if defined but empty, look in dns srv
url = config.get(section, 'wapt_server')
if url:
self._server_url = url
else:
self._server_url = None
else:
# no server at all
self._server_url = ''
if config.has_option(section, 'use_kerberos'):
self.use_kerberos = config.getboolean(section, 'use_kerberos')
if config.has_option(section, 'use_http_proxy_for_server') and config.getboolean(section, 'use_http_proxy_for_server'):
if config.has_option(section, 'http_proxy'):
self.proxies = {'http': config.get(section, 'http_proxy'), 'https': config.get(section, 'http_proxy')}
else:
self.proxies = None
else:
self.proxies = {'http': None, 'https': None}
if config.has_option(section, 'wapt_server_timeout'):
self.timeout = config.getfloat(section, 'wapt_server_timeout')
if config.has_option(section, 'dnsdomain'):
self.dnsdomain = config.get(section, 'dnsdomain')
if config.has_option(section, 'verify_cert'):
self.verify_cert = get_verify_cert(config.get(section, 'verify_cert'))
if config.has_option(section, 'client_certificate') and config.get(section, 'client_certificate'):
self.client_certificate = config.get(section, 'client_certificate')
if config.has_option(section, 'client_private_key') and config.get(section, 'client_private_key'):
self.client_private_key = config.get(section, 'client_private_key')
self.clear_session()
return self
[docs] def load_config_from_file(self, config_filename, section='global'):
"""Load waptserver configuration from an inifile located at config_filename
Args:
config_filename (str) : path to wapt inifile
section (str): ini section from which to get parameters. default to 'global'
Returns:
WaptServer: self
"""
ini = RawConfigParser()
ini.read(config_filename)
self.load_config(ini, section)
return self
[docs] def get(self, action, auth=None, timeout=None, use_ssl_auth=True, enable_capture_external_ip = True, enable_password_callback=True, decode_json=True):
"""Make a get http request to the server, and return result decoded from json
This assuems remo
Args:
action (str): doc part of the url
auth (tuple): authentication passed to requests (user,password)
enable_capture_external_ip (bool) : if true and callback is defined, try to get external IP from X-Remote-IP header
and set it though self.capture_external_ip_callback
enable_password_callback (bool) : if true, password callback will be called if needed.
Returns:
dict : response returned from server as json data.
"""
if self.server_url:
with self.get_requests_session(use_ssl_auth=use_ssl_auth) as session:
req = session.get("%s/%s" % (self.server_url, action),
timeout=timeout or self.timeout,
auth=auth,
allow_redirects=True)
if req.status_code == 401:
req = session.get("%s/%s" % (self.server_url, action),
timeout=timeout or self.timeout,
auth=self.auth(action=action, enable_password_callback=enable_password_callback),
allow_redirects=True)
# if ssl auth has issue, retry without ssl_auth
if req.status_code == 400 and use_ssl_auth:
with self.get_requests_session(use_ssl_auth=False) as session2:
req = session2.get("%s/%s" % (self.server_url, action),
timeout=timeout or self.timeout,
auth=auth,
allow_redirects=True)
if req.status_code == 401:
req = session2.get("%s/%s" % (self.server_url, action),
timeout=timeout or self.timeout,
auth=self.auth(action=action, enable_password_callback=enable_password_callback),
allow_redirects=True)
req.raise_for_status()
if enable_capture_external_ip and req.headers.get('X-Remote-IP') and self.capture_external_ip_callback:
self.capture_external_ip_callback(req.headers['X-Remote-IP'])
if decode_json:
return ujson.loads(req.content)
else:
return req.content
else:
raise EWaptBadSetup('Wapt server url not defined')
[docs] def head(self, action, auth=None, timeout=None, use_ssl_auth=True, enable_capture_external_ip=True, enable_password_callback=True):
""" """
if self.server_url:
with self.get_requests_session(use_ssl_auth=use_ssl_auth) as session:
req = session.head("%s/%s" % (self.server_url, action),
timeout=timeout or self.timeout,
auth=auth,
allow_redirects=True)
if req.status_code == 401:
req = session.head("%s/%s" % (self.server_url, action),
timeout=timeout or self.timeout,
auth=self.auth(action=action, enable_password_callback=enable_password_callback),
allow_redirects=True)
req.raise_for_status()
if enable_capture_external_ip and req.headers.get('X-Remote-IP') and self.capture_external_ip_callback:
self.capture_external_ip_callback(req.headers['X-Remote-IP'])
return req.headers
else:
raise EWaptBadSetup('Wapt server url not defined')
[docs] def post(self, action, data=None, files=None, auth=None, timeout=None, signature=None, signer=None, content_length=None, use_ssl_auth=True, enable_capture_external_ip=True, enable_password_callback=True,max_retry_count=3):
"""Post data to waptserver using http POST method
Add a signature to the posted data using host certificate.
Posted Body is gzipped
Args:
action (str): doc part of the url
data (str) : posted data body
files (list or dict) : list of filenames
Returns:
dict : response returned from server as json data.
"""
if self.server_url:
with self.get_requests_session(use_ssl_auth=use_ssl_auth) as session:
if data:
if action == 'add_host_kerberos':
try:
__, krb_context = kerberos.authGSSClientInit("HTTP@%s" % str(self.server_url).split('//', 1)[1].split(':')[0])
kerberos.authGSSClientStep(krb_context, "")
negotiate_details = kerberos.authGSSClientResponse(krb_context)
session.headers.update({"Authorization": "Negotiate " + negotiate_details})
except:
pass
session.headers.update({
'Content-type': 'binary/octet-stream',
'Content-transfer-encoding': 'binary',
})
if isinstance(data, str):
data = data.encode('utf-8')
if isinstance(data, bytes):
session.headers['Content-Encoding'] = 'gzip'
data = zlib.compress(data)
if signature:
session.headers.update({
'X-Signature': base64.b64encode(signature),
})
if signer:
session.headers.update({
'X-Signer': signer,
})
if content_length is not None:
session.headers['Content-Length'] = "%s" % content_length
if isinstance(files, list):
files_dict = {}
for fn in files:
with open(fn, 'rb') as f:
files_dict[os.path.basename(fn)] = f.read()
elif isinstance(files, dict):
files_dict = files
else:
files_dict = None
# check if auth is required before sending data in chunk
retry_count = 0
if files_dict:
while True:
req = session.head("%s/%s" % (self.server_url, action),
timeout=timeout or self.timeout,
auth=auth,
allow_redirects=True)
if req.status_code == 401:
retry_count += 1
if retry_count >= 3:
raise EWaptBadServerAuthentication('Authentication failed on server %s for action %s' % (self.server_url, action))
auth = self.auth(action=action, enable_password_callback=enable_password_callback)
else:
break
if type(data) is FileChunks:
filechunks = data
data = filechunks.get()
while True:
req = session.post("%s/%s" % (self.server_url, action),
data=data,
files=files_dict,
timeout=timeout or self.timeout,
auth=auth,
allow_redirects=True)
if (req.status_code == 401) and (retry_count < max_retry_count):
retry_count += 1
if retry_count >= 3:
raise EWaptBadServerAuthentication('Authentication failed on server %s for action %s' % (self.server_url, action))
if 'filechunks' in locals():
filechunks.reopen()
data = filechunks.get()
auth = self.auth(action=action, enable_password_callback=enable_password_callback)
else:
break
req.raise_for_status()
# requires that nginx reverse proxy set this 'X-Remote-IP' header.
if enable_capture_external_ip and req.headers.get('X-Remote-IP') and self.capture_external_ip_callback:
self.capture_external_ip_callback(req.headers['X-Remote-IP'])
return ujson.loads(req.content)
else:
raise EWaptBadSetup('Wapt server url not defined')
[docs] def client_auth(self):
"""Return SSL pair (cert,key) filenames for client side SSL auth
Returns:
tuple: (cert path,key path,strkeypassword)
"""
if self.client_certificate and os.path.isfile(self.client_certificate):
if self.client_private_key is None:
cert = SSLCertificate(self.client_certificate)
key = cert.matching_key_in_dirs(password_callback=self.get_private_key_password)
self.client_private_key = key.private_key_filename
return (self.client_certificate, self.client_private_key, self.get_private_key_password(self.server_url, self.client_certificate))
else:
return None
[docs] def available(self):
if self.server_url:
with self.get_requests_session() as session:
try:
req = session.head("%s/ping" % (self.server_url),
timeout=self.timeout,
auth=None,
allow_redirects=True)
if req.status_code == 401:
req = session.head("%s/ping" % (self.server_url),
timeout=self.timeout,
auth=self.auth(action='ping'),
allow_redirects=True)
# try without ssl_auth (self signed client cert)
if req.status_code == 400:
with self.get_requests_session(use_ssl_auth=False) as session2:
req = session2.head("%s/ping" % (self.server_url),
timeout=self.timeout,
auth=None,
allow_redirects=True)
if req.status_code == 401:
req = session2.head("%s/ping" % (self.server_url),
timeout=self.timeout,
auth=self.auth(action='ping'),
allow_redirects=True)
req.raise_for_status()
return True
except Exception as e:
#logger.debug(traceback.format_exc())
logger.debug('Wapt server %s unavailable (%s)' % (self._server_url, e))
return False
else:
logger.debug('Wapt server is unavailable because no URL is defined')
return False
[docs] def as_dict(self):
result = {}
attributes = ['server_url', 'proxies', 'dnsdomain']
for att in attributes:
result[att] = getattr(self, att)
return result
[docs] def ask_user_password(self, action=None):
"""Ask for basic auth if server requires it"""
if self.ask_user_password_hook is not None:
return self.ask_user_password_hook(action) # pylint: disable=not-callable
elif self.interactive_session:
user = input('Please provide username for action "%s" on server %s: ' % (action, self.server_url))
if user:
password = getpass.getpass('Password: ')
if user and password:
return (ensure_unicode(user).encode('utf8'), ensure_unicode(password).encode('utf8'))
else:
return None
else:
return None
def __repr__(self):
try:
return '<WaptServer %s verify_cert=%s client_cert_path=%s>' % (self.server_url, self.verify_cert, self.client_certificate)
except:
return '<WaptServer %s>' % 'unknown'
[docs]class WaptRepo(WaptRemoteRepo):
"""Gives access to a remote http repository, with a zipped Packages packages index
Find its repo_url based on
* repo_url explicit setting in ini config section [<name>]
* if there is some rules use rules
>>> repo = WaptRepo(name='main',url='http://wapt/wapt',timeout=4)
>>> packages = repo.packages()
>>> len(packages)
"""
def __init__(self, url=None, name='wapt', verify_cert=None, http_proxy=None, timeout=None, cabundle: [SSLCABundle,Callable[[],SSLCABundle]]=None, config:RawConfigParser=None, section=None, WAPT:"Wapt"=None):
"""Initialize a repo at url "url".
Args:
name (str): internal local name of this repository
url (str): http URL to the repository.
If url is None, the url is requested at the server.
http_proxy (str): URL to http proxy or None if no proxy.
timeout (float): timeout in seconds for the connection to the rmeote repository
wapt_server (str): WAPT Server URL to use for autodiscovery if url is not supplied.
.. versionchanged:: 1.4.0
authorized_certs (list): list of trusted SSL certificates to filter out untrusted entries.
if None, no check is performed. All antries are accepted.
.. versionchanged:: 1.5.0
cabundle (SSLCABundle or callable which returns a SSLCABundle like Wapt.get_cabundle): list of trusted SSL ca certificates to filter out untrusted entries.
if None, no check is performed. All antries are accepted.
"""
self._WAPT = None
self.WAPT = WAPT
# create additional properties
self._rules = None
self._cached_wapt_repo_url = None
self._repo_url_already_calculated = False
self._cached_http_proxy = None
WaptRemoteRepo.__init__(self, url=url, name=name, verify_cert=verify_cert, http_proxy=http_proxy, timeout=timeout, cabundle=cabundle, config=config, section=section)
[docs] def reset_network(self):
"""called by wapt when network configuration has changed"""
self._rules = None
self._cached_wapt_repo_url = None
self._repo_url_already_calculated = False
self._cached_http_proxy = None
self._packages = None
self._packages_date = None
@property
def WAPT(self):
return self._WAPT
@WAPT.setter
def WAPT(self, value):
if value != self.WAPT:
self._WAPT = value
[docs] def rulesdb(self):
"""
Get rules from DB
"""
if self.name in ('wapt', 'wapt-host', 'waptwua'):
if self.WAPT is not None:
rules = self.WAPT.waptdb.get_param('repo_rules-wapt')
return rules if isinstance(rules, list) else []
return []
@property
def rules(self):
if self._rules is None:
all_rules = self.rulesdb()
self._rules = []
for rule in all_rules:
if self.name in rule['repositories']:
self._rules.append(rule)
return self._rules
@property
def cached_wapt_repo_url(self):
if self._repo_url_already_calculated:
return self._cached_wapt_repo_url
else:
return self.find_wapt_repo_url() if self.WAPT is not None and self.WAPT.use_repo_rules and self.rules else None
@cached_wapt_repo_url.setter
def cached_wapt_repo_url(self, value):
if value != self._cached_wapt_repo_url:
if value:
value = value.rstrip('/')
self._cached_wapt_repo_url = value
@property
def repo_url(self):
"""Repository URL
Fixed url if none is set in wapt-get.ini by querying the server.
The URL is queried once and then cached into a local property.
Returns:
str: url to the repository
>>> repo = WaptRepo(name='wapt',timeout=4)
>>> print repo.wapt_server
http://wapt.wapt.fr/
>>> repo = WaptRepo(name='wapt',timeout=4)
>>> print repo.wapt_server
http://wapt.wapt.fr/
>>> print repo.repo_url
http://srvwapt.tranquilit.local/wapt
"""
calculated_repo = self.cached_wapt_repo_url
return calculated_repo if calculated_repo else self._repo_url
@repo_url.setter
def repo_url(self, value):
if value:
value = value.rstrip('/')
if value != self._repo_url:
self.reset_network()
self._repo_url = value
@property
def proxies(self):
"""dict for http proxies url suitable for requests based on the http_proxy repo attribute
Returns:
dict: {'http':'http://proxy:port','https':'http://proxy:port'}
"""
if self._cached_http_proxy:
proxy = self._cached_http_proxy
elif self.http_proxy:
proxy = self.http_proxy
else:
proxy = None
return {'http': proxy, 'https': proxy}
[docs] def find_wapt_repo_url(self):
"""Find a wapt_repo_url from rules
Returns:
str: URL to the repo.
"""
def rule_agent_ip(rule):
try:
ip_network = ipaddress.ip_network(rule['value'])
except ValueError:
return False
for ip in get_main_ip(urllib.parse.urlparse(rule['repo_url']).netloc):
if ipaddress.ip_address(ip) in ip_network:
return True
return False
def rule_domain(rule):
return setuphelpers.get_domain().lower() == rule['value'].lower()
def rule_hostname(rule):
return fnmatch.fnmatch(setuphelpers.get_hostname().lower(), rule['value'].lower())
def rule_public_ip(rule):
ip = self.WAPT.waptdb.get_param('last_external_ip')
return ip and (ipaddress.ip_address(ip) in ipaddress.ip_network(rule['value']))
def rule_site(rule):
return self.WAPT.get_host_site().lower() == rule['value'].lower()
def check_rule(rule_condition, rule):
return {
'AGENT IP': rule_agent_ip,
'DOMAIN': rule_domain,
'HOSTNAME': rule_hostname,
'PUBLIC IP': rule_public_ip,
'SITE': rule_site
}[rule_condition](rule)
self._repo_url_already_calculated = True
for rule in sorted(self.rules, key=itemgetter('sequence')):
try:
if (not(rule.get('negation', False)) == check_rule(rule['condition'], rule)) and \
(rule.get('no_fallback', False) or \
((self.name == 'waptwua' and ('download.windowsupdate.com' in rule['repo_url'])) or \
super(WaptRepo, self).is_available(url=rule['repo_url'], http_proxy=rule['http_proxy'] if rule.get('has_proxy', False) else None) is not None)):
self.cached_wapt_repo_url = rule['repo_url'].rstrip('/')+'-host' if isinstance(self, WaptHostRepo) else rule['repo_url']
rule['active_rule'] = True
self._cached_http_proxy = rule['http_proxy'] if rule.get('has_proxy', False) else None
return self.cached_wapt_repo_url
except Exception as e:
logger.critical("The rule %s failed for repo %s with repo_url %s : %s" % (rule['name'], self.name, rule['repo_url'], str(e)))
rule['exception'] = str(e)
self.cached_wapt_repo_url = None
self._cached_http_proxy = None
return None
[docs] def load_config(self, config, section=None):
"""Load waptrepo configuration from inifile section.
Use name of repo as section name if section is not provided.
Use 'global' if no section named section in ini file
"""
if not section:
section = self.name
# creates a default parser with a default section if None provided to get defaults
if config is None:
config = RawConfigParser(self._default_config)
config.add_section(section)
if not config.has_section(section):
section = 'global'
if config.has_option(section, 'repo_url'):
self._repo_url = config.get(section, 'repo_url')
elif config.has_option('global', 'repo_url'):
self._repo_url = config.get('global', 'repo_url')
WaptRemoteRepo.load_config(self, config, section)
return self
[docs] def as_dict(self):
result = super(WaptRepo, self).as_dict()
result.update(
{
'repo_url': self.repo_url,
'rules': self._rules,
})
return result
def __repr__(self):
try:
return '<WaptRepo %s verify_cert=%s client_cert_path=%s>' % (self.repo_url, self.verify_cert, self.client_certificate)
except:
return '<WaptRepo unknown>'
[docs]class WaptHostRepo(WaptRepo):
"""Dummy http repository for host packages
>>> host_repo = WaptHostRepo(name='wapt-host',host_id=['0D2972AC-0993-0C61-9633-529FB1A177E3','4C4C4544-004E-3510-8051-C7C04F325131'])
>>> host_repo.load_config_from_file(r'C:\\Users\htouvet\AppData\Local\waptconsole\waptconsole.ini')
>>> host_repo.packages()
[PackageEntry('0D2972AC-0993-0C61-9633-529FB1A177E3','10') ,
PackageEntry('4C4C4544-004E-3510-8051-C7C04F325131','30') ]
"""
def __init__(self, url=None, name='wapt-host', verify_cert=None, http_proxy=None, timeout=None, host_id=None, cabundle=None, config=None, section=None, host_key=None, WAPT=None):
self._host_id = None
self.host_key = None
WaptRepo.__init__(self, url=url, name=name, verify_cert=verify_cert, http_proxy=http_proxy, timeout=timeout, cabundle=cabundle, config=config, section=section, WAPT=WAPT)
self.host_id = host_id
if host_key:
self.host_key = host_key
[docs] def host_package_url(self, host_id=None):
if host_id is None:
if self.host_id and isinstance(self.host_id, list):
host_id = self.host_id[0]
else:
host_id = self.host_id
return "%s/%s.wapt" % (self.repo_url, host_id)
[docs] def is_available(self):
logger.debug('Checking availability of %s' % (self.name))
try:
host_package_url = self.host_package_url()
with self.get_requests_session() as session:
logger.debug('Trying to get host package for %s at %s' % (self.host_id, host_package_url))
req = session.head(host_package_url,
timeout=self.timeout,
allow_redirects=True)
req.raise_for_status()
packages_last_modified = req.headers.get('last-modified')
return httpdatetime2isodate(packages_last_modified)
except requests.HTTPError:
logger.info('No host package available at this time for %s on %s' % (self.host_id, self.name))
return None
[docs] def load_config(self, config, section=None):
"""Load waptrepo configuration from inifile section.
Use name of repo as section name if section is not provided.
Use 'global' if no section named section in ini file
"""
if not section:
section = self.name
# creates a default parser with a default section if None provided to get defaults
if config is None:
config = RawConfigParser(self._default_config)
config.add_section(section)
if not config.has_section(section):
if config.has_section('wapt-main'):
section = 'wapt-main'
else:
section = 'global'
WaptRepo.load_config(self, config, section)
return self
@property
def repo_url(self):
# hack to get implicit repo_url from main repo_url
repo_url = super(WaptHostRepo, self).repo_url
if repo_url and self._section in ['wapt-main', 'global'] and not repo_url.endswith('-host'):
return repo_url+'-host'
else:
return repo_url
@repo_url.setter
def repo_url(self, value):
if value:
value = value.rstrip('/')
if value != self._repo_url:
self.reset_network()
self._repo_url = value
@property
def host_id(self):
return self._host_id
@host_id.setter
def host_id(self, value):
if value != self._host_id:
self._packages = None
self._packages_date = None
self._index = {}
self._index_by_uuid = {}
self._host_id = value
def _load_packages_index(self):
self._packages = []
self._index = {}
self._index_by_uuid = {}
self.discarded = []
if not self.repo_url:
raise EWaptException('URL for WaptHostRepo repository %s is empty. Either add a wapt-host section in ini, or add a correct wapt_server and rules' % (self.name))
if self.host_id and not isinstance(self.host_id, list):
host_ids = [self.host_id]
else:
host_ids = self.host_id
with self.get_requests_session() as session:
for host_id in host_ids:
host_package_url = self.host_package_url(host_id)
logger.debug('Trying to get host package for %s at %s' % (host_id, host_package_url))
host_package = session.get(host_package_url,
timeout=self.timeout,
allow_redirects=True,
)
# prepare a package entry for further check
package = PackageEntry()
package.package = host_id
package.repo = self.name
package.repo_url = self.repo_url
if host_package.status_code == 404:
# host package not found
logger.info('No host package found for %s' % host_id)
package._packages_date = '1900-01-01T00:00:00'
package._package_content = None
self._packages_date = package._packages_date
else:
# for other than not found error, add to the discarded list.
# this can be consulted for mass changes to not recreate host packages because of temporary failures
try:
host_package.raise_for_status()
except requests.HTTPError as e:
logger.info('Discarding package for %s: error %s' % (package.package, e))
self.discarded.append(package)
continue
content = host_package.content
if not content.startswith(zipfile.stringFileHeader):
# try to decrypt package data
if self.host_key:
_host_package_content = self.host_key.decrypt_fernet(content)
else:
raise EWaptNotAPackage('Package for %s does not look like a Zip file and no key is available to try to decrypt it' % host_id)
else:
_host_package_content = content
# Packages file is a zipfile with one Packages file inside
with CustomZipFile(io.BytesIO(_host_package_content)) as zip:
control_data = codecs.decode(zip.read(name='WAPT/control'), 'UTF-8')
package._load_control(control_data)
package.filename = package.make_package_filename()
try:
cert_data = zip.read(name='WAPT/certificate.crt')
signers_bundle = SSLCABundle()
signers_bundle.add_certificates_from_pem(cert_data)
except Exception as e:
logger.warning('Error reading host package certificate: %s' % repr(e))
signers_bundle = None
if self.is_locally_allowed_package(package):
try:
if self.cabundle is not None:
package.check_control_signature(self.cabundle, signers_bundle=signers_bundle)
self._add_package(package)
# keep content with index as it should be small
package._package_content = _host_package_content
package._packages_date = httpdatetime2isodate(host_package.headers.get('last-modified', None))
# TODO better
self._packages_date = package._packages_date
except (SSLVerifyException, EWaptNotSigned) as e:
logger.critical("Control data of package %s on repository %s is either corrupted or doesn't match any of the expected certificates %s" % (package.asrequirement(), self.name, self.cabundle))
logger.debug("%s: %s" % (package.asrequirement(),e))
self.discarded.append(package)
else:
logger.info('Discarding %s on repo "%s" because of local whitelist/blacklist rules' % (package.asrequirement(), self.name))
self.discarded.append(package)
[docs] def download_packages(self, package_requests, target_dir=None, usecache=True, printhook=None):
"""Download a list of packages from repo
Args:
package_request (list,PackageEntry): a list of PackageEntry to download
target_dir (str): where to store downloaded Wapt Package files
usecache (bool): wether to try to use cached Wapt files if checksum is ok
printhook (callable): to show progress of download
Returns:
dict: {"downloaded":[local filenames],"skipped":[filenames in cache],"errors":[],"packages":self.packages()}
"""
if not isinstance(package_requests, (list, tuple)):
package_requests = [package_requests]
if not target_dir:
target_dir = tempfile.mkdtemp()
downloaded = []
errors = []
self._load_packages_index()
# if multithread... we don't have host package in memory cache from last self._load_packages_index
for pr in package_requests:
for pe in self.packages():
if ((isinstance(pr, PackageEntry) and (pe == pr)) or
(isinstance(pr, str) and pe.match(pr))):
pfn = os.path.join(target_dir, pe.make_package_filename())
if not pfn.endswith('.wapt'):
raise EWaptNotAPackage('The file %s does not have a .wapt extension' % pfn)
if pe._package_content is not None:
with open(pfn, 'wb') as package_zip:
package_zip.write(pe._package_content)
pe.localpath = pfn
# for further reference
if isinstance(pr, PackageEntry):
pr.localpath = pfn
downloaded.append(pfn)
if not os.path.isfile(pfn):
logger.warning('Unable to write host package %s into %s' % (pr.asrequirement(), pfn))
errors.append(pfn)
else:
logger.warning('No host package content for %s' % (pr.asrequirement(),))
break
return {"downloaded": downloaded, "skipped": [], "errors": [], "packages": self.packages()}
def __repr__(self):
return '<WaptHostRepo %s for host_id %s >' % (self.repo_url, self.host_id)
[docs]class WaptPackageInstallLogger(LogOutput):
"""Context handler to log all print messages to a wapt package install log
Args:
wapt_context (Wapt): Wapt instance
package_name (str): name of running or installed package local status where to log status and output
>>>
"""
def __init__(self, console, wapt_context=None, install_id=None, user=None, running_status='RUNNING', exit_status='OK', error_status='ERROR'):
self.wapt_context = wapt_context
self.install_id = install_id
self.last_stdout_line = ''
self.user = user
if self.user is None:
self.user = setuphelpers.get_current_user()
def update_install_status(append_line=None, set_status=None, context=None):
if self.wapt_context:
self.wapt_context.update_package_install_status(
rowid=context.install_id,
set_status=set_status,
append_line=append_line)
self.wapt_context.runstatus = append_line or ''
if append_line and hasattr(self.wapt_context, 'events') and self.wapt_context.events:
self.wapt_context.events.post_event('PRINT', ensure_unicode(append_line))
LogOutput.__init__(self, console=console,
update_status_hook=update_install_status,
context=self,
running_status=running_status,
exit_status=exit_status,
error_status=error_status)
[docs]class WaptPackageSessionSetupLogger(LogOutput):
"""Context handler to log all print messages to a wapt package install log
Args:
wapt_context (Wapt): Wapt instance
package_name (str): name of running or installed package local status where to log status and output
>>>
"""
def __init__(self, console, waptsessiondb, install_id, running_status='RUNNING', exit_status=None, error_status='ERROR'):
self.waptsessiondb = waptsessiondb
self.install_id = install_id
def update_install_status(append_line=None, set_status=None, context=None):
self.waptsessiondb.update_install_status(
rowid=context.install_id,
set_status=set_status,
append_line=append_line)
LogOutput.__init__(self, console=console,
update_status_hook=update_install_status,
context=self,
running_status=running_status,
exit_status=exit_status,
error_status=error_status)
[docs]class WaptPackageAuditLogger(LogOutput):
"""Context handler to log all print messages to a wapt package audit log
Args:
console (file) : sys.stderr
wapt_context (Wapt): Wapt instance
install_id (int): name of running or installed package local status where to log status and output
>>>
"""
def __init__(self, console, wapt_context=None, install_id=None, user=None, running_status='RUNNING', exit_status=None, error_status='ERROR'):
self.wapt_context = wapt_context
self.install_id = install_id
self.user = user
if self.user is None:
self.user = setuphelpers.get_current_user()
def update_audit_status(append_line=None, set_status=None, context=None):
self.wapt_context.waptdb.update_audit_status(
rowid=context.install_id,
set_status=set_status,
append_line=append_line)
LogOutput.__init__(self, console=console,
update_status_hook=update_audit_status,
context=self,
running_status=running_status,
exit_status=exit_status,
error_status=error_status)
######################
wapt_count = [0,]
[docs]class Wapt(BaseObjectClass):
"""Global WAPT engine"""
global_attributes = ['wapt_base_dir', 'waptserver', 'config_filename', 'proxies', 'repositories', 'personal_certificate_path',
'public_certs_dir', 'packages_cache_dir', 'dbpath',
'http_proxy','use_http_proxy_for_repo','use_http_proxy_for_server',
'limit_bandwidth',
'waptservice_admin_filter','waptservice_port','waptservice_poll_timeout',
'locales','custom_tags','packages_whitelist','packages_blacklist','maturities',
'host_uuid','use_fqdn_as_uuid','use_hostpackages','use_ad_groups','use_repo_rules',
'host_profiles','host_organizational_unit_dn','host_ad_site',
'allow_user_service_restart','allow_remote_shutdown','allow_remote_reboot',
'ldap_auth_server','ldap_auth_base_dn','ldap_auth_ssl_enabled','verify_cert_ldap',
'loglevel','loglevel_waptcore','loglevel_waptservice','loglevel_wapttasks','loglevel_waptws','loglevel_waptdb','loglevel_websocket','loglevel_waitress','log_to_windows_events',
'download_after_update_with_waptupdate_task_period',
'websockets_ping','websockets_retry_delay','websockets_check_config_interval','websockets_hurry_interval',
'notify_user',
'waptaudit_task_period',
'signature_clockskew',
'wol_relay',
'hiberboot_enabled','max_gpo_script_wait','pre_shutdown_timeout',
'minimum_battery_percent',
'uninstallkey_timeout',
'check_certificates_validity',
'token_lifetime',
'repositories',
'trust_all_certs_in_pems',
'wapt_temp_dir',
'peercache_enable',
]
_default_config = {
'loglevel': 'warning',
'log_to_windows_events': '0',
'use_http_proxy_for_repo': '0',
'use_http_proxy_for_server': '0',
'tray_check_interval': 2,
'use_hostpackages': '1',
'use_ad_groups': '0',
'timeout': 10.0,
'wapt_server_timeout': 30.0,
'maturities': 'PROD',
'default_maturity': '',
'http_proxy': '',
'token_lifetime': 24*60*60, # 24 hours
'trust_all_certs_in_pems': '0',
# optional...
'default_sources_root': 'c:\\waptdev' if (os.name == 'nt') else os.path.join(os.path.expanduser('~'), 'waptdev'),
'default_package_prefix': 'tis',
'default_sources_suffix': 'wapt',
'default_sources_url': '',
'upload_cmd': '',
'upload_cmd_host': '',
'after_upload': '',
'personal_certificate_path': '',
'check_certificates_validity': '1',
'use_fqdn_as_uuid': '0',
'uninstallkey_timeout': 120,
}
public_params = ['server_uuid','uuid','hostname','hardware_uuid','host_organizational_unit_dn','licences','domain_info','last_domain_info_date','host_description','last_successful_register']
def __init__(self, config_filename=None, defaults=None, disable_update_server_status=True, wapt_base_dir=None, dbpath=None, publicdbpath=None, merge_config_packages=None):
"""Initialize engine.
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> updates = wapt.update()
>>> 'count' in updates and 'added' in updates and 'upgrades' in updates and 'date' in updates and 'removed' in updates
True
"""
# used to signal to cancel current operations ASAP
global wapt_count
wapt_count[0] += 1
#print(wapt_count[0])
self.task_is_cancelled = threading.Event()
assert not config_filename or isinstance(config_filename, str)
self._threadid = threading.get_ident()
self.lock=threading.Lock()
self._waptdb = None
self._waptpublicdb = None
self._waptsessiondb = None
self._dbpath = dbpath
self._publicdbpath = publicdbpath
# cached runstatus to avoid setting in db if not changed.
self._runstatus = None
if wapt_base_dir:
self.wapt_base_dir = wapt_base_dir
else:
self.wapt_base_dir = os.path.dirname(os.path.abspath(waptutils__file__))
self.config: RawConfigParser = None
self.config_filename = config_filename
if not self.config_filename:
self.config_filename = self.default_config_filename()
self.private_dir = os.path.join(self.wapt_base_dir, 'private')
self.public_dir = os.path.join(self.wapt_base_dir, 'public')
self.persistent_root_dir = os.path.join(self.private_dir, 'persistent')
self.public_persistent_root_dir = os.path.join(self.public_dir,'persistent')
self.token_lifetime = 24*60*60
self.disable_update_server_status = disable_update_server_status
self.configs_dir = os.path.join(self.wapt_base_dir, 'conf.d')
self.load_config(config_filename=self.config_filename, merge_config_packages=merge_config_packages)
self.options = OptionParser()
self.options.force = False
self._last_date_wmi_send = None
self._last_date_dmi_send = None
# list of process pids launched by run command
self.pidlist = []
# events handler
self.events = None
self.progress_hook = None
if sys.platform == 'win32':
pythoncom.CoInitialize()
[docs] def config_parser(self):
logger.debug('New INI Parser')
config = RawConfigParser(defaults=self._default_config)
if os.path.isfile(self.config_filename):
config.read(self.config_filename)
return config
[docs] def reset_settings(self):
self.config=None
self._host_uuid = None
self._merge_config_packages = True
self._public_certs_dir = None
self.use_hostpackages = True
self.use_ad_groups = False
self.waptaudit_task_period = "2h"
self.peercache_enable = False
self._repositories = None
self._wua_repository = None
self.upload_cmd = None
self.upload_cmd_host = self.upload_cmd
self.after_upload = None
self.proxies = None
self.language = setuphelpers.get_language()
self.locales = [setuphelpers.get_language()]
self.maturities = ['PROD', '']
# default maturity when importing or creating new package
self.default_maturity = ''
self.filter_on_host_cap = True
self.use_http_proxy_for_repo = False
self.use_http_proxy_for_server = False
self.forced_uuid = None
self.use_fqdn_as_uuid = False
# to redirect stdout
self.redirect_stdout_to = None
# where to pre-download the packages
self.packages_cache_dir = None
# where to unzip packages for installation
self.wapt_temp_dir = None
# to allow/restrict installation, supplied to packages
self.user = setuphelpers.get_current_user()
self.usergroups = None
# host key cache
self._host_key = None
self._host_key_timestamp = None
self._host_certificate = None
self._host_certificate_timestamp = None
# for private key password dialog tales (location,indentity) parameters
self._private_key_password_callback = None
# keep private key in cache
self._private_key_cache = None
self._cabundle = None
self.check_certificates_validity = False
self._waptserver = None
self.packages_whitelist = None
self.packages_blacklist = None
self._host_profiles = None
self.use_repo_rules = False
# if True: trust all certificates in each PEM file of wapt/ssl
# if False: trust only the first certificate of each file.
self.trust_all_certs_in_pems = False
@property
def public_certs_dir(self)->str:
if self._public_certs_dir is None:
self._public_certs_dir = os.path.join(self.wapt_base_dir, 'ssl')
return self._public_certs_dir
@public_certs_dir.setter
def public_certs_dir(self, value:str):
self._public_certs_dir = value
self._cabundle = None
[docs] def get_cabundle(self)->SSLCABundle:
# lazy loading to handle config changes
if self._cabundle is None:
self._cabundle = SSLCABundle()
self._cabundle.add_pems(self.public_certs_dir,trust_first=True,trust_all=self.trust_all_certs_in_pems,load_keys=False)
return self._cabundle
@property
def cabundle(self)->SSLCABundle:
return self.get_cabundle()
@property
def merged_config_hash(self)->str:
return self.read_param('merged_config_hash')
@merged_config_hash.setter
def merged_config_hash(self,value):
if self.read_param('merged_config_hash') != value:
return self.write_param('merged_config_hash',value)
[docs] def get_config_files_list(self):
r = [d for d in (self.config_filename, self.configs_dir, self.public_certs_dir, self.get_host_certificate_filename()) if d and (os.path.isfile(d) or os.path.isdir(d))]
r.extend([os.path.join(self.configs_dir,f) for f in os.listdir(self.configs_dir)])
return r
[docs] def get_config_hash(self):
return get_files_timestamp_sha256(self.get_config_files_list())
[docs] def default_config_filename(self):
return os.path.join(self.wapt_base_dir, 'wapt-get.ini')
@property
def private_key_password_callback(self):
return self._private_key_password_callback
@private_key_password_callback.setter
def private_key_password_callback(self, value):
self._private_key_password_callback = value
if self._waptserver:
self._waptserver.private_key_password_callback = value
if self._repositories:
for repo in self._repositories:
repo.private_key_password_callback = value
def __enter__(self):
#self.lock.acquire(timeout=30)
return self
def __exit__(self, type, value, tb):
#self.lock.release()
pass
[docs] def as_dict(self):
result = {}
for att in self.global_attributes:
if hasattr(self, att):
result[att] = getattr(self, att)
return result
@property
def dbpath(self):
if self._waptdb:
return self._waptdb.dbpath
if not self._dbpath:
if self.config.has_option('global', 'dbpath') and self.config.get('global', 'dbpath') != '':
self._dbpath = self.config.get('global', 'dbpath')
else:
self._dbpath = os.path.join(self.private_dir, 'db', 'waptdb.sqlite')
# fallback for old wapt
if not os.path.isfile(self._dbpath) and os.path.isfile(os.path.join(self.wapt_base_dir, 'db', 'waptdb.sqlite')):
self._dbpath = os.path.join(self.wapt_base_dir, 'db', 'waptdb.sqlite')
return self._dbpath
@dbpath.setter
def dbpath(self, value):
# check if not changed
if self._waptdb and self._waptdb.dbpath == value:
return
# updated : reset db
self._waptdb = None
self._dbpath = value
@property
def publicdbpath(self):
if self._waptpublicdb:
return self._waptpublicdb.dbpath
if not self._publicdbpath:
self._publicdbpath = os.path.join(self.wapt_base_dir,'db','waptpublicdb.sqlite')
return self._publicdbpath
@publicdbpath.setter
def publicdbpath(self, value):
# check if not changed
if self._waptpublicdb and self._waptpublicdb.dbpath == value:
return
# updated : reset db
self._waptpublicdb = None
self._publicdbpath = value
@property
def host_profiles(self):
result = []
if self._host_profiles is not None:
result.extend(self._host_profiles)
if self.use_ad_groups:
result.extend(self.get_cache_domain_info()['groups'])
return result
[docs] def set_client_cert_auth(self, connection, force=False):
"""Set client side ssl authentication for a waptserver or a waptrepo using
host_certificate if client_certificate is not yet set in config and host certificate is able to do client_auth
Args:
connection: object with client_certificate, client_private_key and client_auth
"""
try:
# use implicit host client certificate if not already set by config
if force or connection.client_certificate is None:
if os.path.isfile(self.get_host_certificate_filename()) and os.path.isfile(self.get_host_key_filename()):
crt = self.get_host_certificate()
if crt.is_client_auth:
logger.debug('Using host certificate %s for repo %s auth' % (self.get_host_key_filename(), connection.name))
connection.client_certificate = self.get_host_certificate_filename()
connection.client_private_key = self.get_host_key_filename()
else:
logger.warning('Host client certificate %s has not client_auth capability, not using it for auth on %s' % (self.get_host_certificate_filename(), connection.name))
else:
logger.debug('Host certificate %s not found, not using it for auth on repo %s' % (self.get_host_certificate_filename(), connection.name))
connection.private_key_password_callback = self.private_key_password_callback
except Exception as e:
logger.debug('Unable to use client certificate auth: %s' % e)
[docs] def save_external_ip(self, ip):
self.write_param('last_external_ip', ip)
[docs] def save_last_domain_info_date(self, lastdate):
self.write_param('last_domain_info_date', lastdate, public=True)
[docs] def save_domain_info(self, info):
self.write_param('domain_info', info, public=True)
[docs] def load_config(self, config_filename=None, merge_config_packages=None):
"""Load configuration parameters from supplied inifilename
"""
self.reset_settings()
if merge_config_packages is None:
merge_config_packages = self._merge_config_packages
# default config file
if not self.config:
self.config = RawConfigParser(defaults=self._default_config)
if config_filename:
self.config_filename = config_filename
if self.config_filename:
relative_configs_dir = os.path.join(os.path.dirname(self.config_filename),'conf.d')
if os.path.isdir(relative_configs_dir):
self.configs_dir = relative_configs_dir
if merge_config_packages and self.configs_dir:
update_ini_from_json_config(self.config_filename,self.configs_dir)
if os.path.isfile(self.config_filename):
with open(self.config_filename,'r',encoding='utf8') as f:
self.config.readfp(f)
# lazzy loading
self._repositories = None
if self.config.has_option('global','wapt_base_dir') and self.config.get('global', 'wapt_base_dir') != '':
self.wapt_base_dir = self.config.get('global','wapt_base_dir')
if self.config.has_option('global', 'private_dir') and self.config.get('global', 'private_dir') != '':
self.private_dir = self.config.get('global', 'private_dir')
else:
self.private_dir = os.path.join(self.wapt_base_dir, 'private')
if self.config.has_option('global', 'public_dir') and self.config.get('global', 'public_dir') != '':
self.public_dir = self.config.get('global', 'public_dir')
else:
self.public_dir = os.path.join(self.wapt_base_dir, 'public')
if self.config.has_option('global', 'persistent_root_dir') and self.config.get('global', 'persistent_root_dir') != '':
self.persistent_root_dir = self.config.get('global', 'persistent_root_dir')
else:
self.persistent_root_dir = os.path.join(self.private_dir, 'persistent')
if self.config.has_option('global', 'public_persistent_root_dir') and self.config.get('global', 'public_persistent_root_dir') != '':
self.public_persistent_root_dir = self.config.get('global', 'public_persistent_root_dir')
else:
self.public_persistent_root_dir = os.path.join(self.public_dir, 'persistent')
if self.config.has_option('global', 'packages_cache_dir') and self.config.get('global', 'packages_cache_dir') != '':
self.packages_cache_dir = self.config.get('global', 'packages_cache_dir')
else:
self.packages_cache_dir = os.path.join(self.private_dir, 'cache')
if self.config.has_option('global', 'wapt_temp_dir'):
self.wapt_temp_dir = self.config.get('global', 'wapt_temp_dir')
if self.config.has_option('global', 'uuid'):
self.forced_uuid = self.config.get('global', 'uuid')
else:
# force reset to None if config file is changed at runtime
self.forced_uuid = None
if self.config.has_option('global', 'use_fqdn_as_uuid'):
self.use_fqdn_as_uuid = self.config.getboolean('global', 'use_fqdn_as_uuid')
if self.config.has_option('global', 'uninstallkey_timeout'):
self.uninstallkey_timeout = self.config.getint('global', 'uninstallkey_timeout')
# must have a matching key either in same file or in same directory
# see self.private_key()
if self.config.has_option('global', 'personal_certificate_path'):
self.personal_certificate_path = self.config.get('global', 'personal_certificate_path')
# be smart with old config
if not self.personal_certificate_path and self.config.has_option('global', 'private_key'):
pk = self.config.get('global', 'private_key')
if pk and os.path.isfile(pk):
(root, ext) = os.path.splitext(pk)
if os.path.isfile(root+'.crt'):
self.personal_certificate_path = root+'.crt'
if self.config.has_option('global', 'public_certs_dir') and self.config.get('global', 'public_certs_dir') != '':
self.public_certs_dir = self.config.get('global','public_certs_dir')
self._cabundle = None
self.trust_all_certs_in_pems = False
# set this to True for backward compatibility.
# but it's more secure to only trust first.
if self.config.has_option('global', 'trust_all_certs_in_pems'):
self.trust_all_certs_in_pems = self.config.getboolean('global', 'trust_all_certs_in_pems')
if self.config.has_option('global', 'check_certificates_validity'):
self.check_certificates_validity = self.config.getboolean('global', 'check_certificates_validity')
if self.config.has_option('global', 'upload_cmd'):
self.upload_cmd = self.config.get('global', 'upload_cmd')
if self.config.has_option('global', 'upload_cmd_host'):
self.upload_cmd_host = self.config.get('global', 'upload_cmd_host')
if self.config.has_option('global', 'after_upload'):
self.after_upload = self.config.get('global', 'after_upload')
self.use_http_proxy_for_repo = self.config.getboolean('global', 'use_http_proxy_for_repo')
self.use_http_proxy_for_server = self.config.getboolean('global', 'use_http_proxy_for_server')
if self.config.has_option('global', 'http_proxy'):
self.proxies = {'http': self.config.get('global', 'http_proxy'), 'https': self.config.get('global', 'http_proxy')}
else:
self.proxies = None
# force reset to None if config file is changed at runtime
self._waptserver = None
if self.config.has_option('global', 'language'):
self.language = self.config.get('global', 'language')
# for testing
if self.config.has_option('global', 'fake_hostname'):
self._set_fake_hostname(self.config.get('global', 'fake_hostname'))
# allow to fake a host Organizational Unit when the computer is not part of an AD, but we want to put host in a OU.
if self.config.has_option('global', 'host_organizational_unit_dn'):
forced_host_organizational_unit_dn = self.config.get('global', 'host_organizational_unit_dn')
if forced_host_organizational_unit_dn != self.host_organizational_unit_dn:
logger.info('Forced forced_host_organizational_unit_dn DB %s' % forced_host_organizational_unit_dn)
self.host_organizational_unit_dn = forced_host_organizational_unit_dn
else:
# force reset to None if config file is changed at runtime
try:
del(self.host_organizational_unit_dn)
except Exception as e:
logger.warning("Error removing host_organizational_unit_dn from db: %s" % e)
# error writing to db because of write access ?
logger.warning('forced OU DN in local wapt db is not matching wapt-get.ini value')
if self.config.has_option('global', 'packages_whitelist'):
self.packages_whitelist = ensure_list(self.config.get('global', 'packages_whitelist'), allow_none=True)
if self.config.has_option('global', 'packages_blacklist'):
self.packages_blacklist = ensure_list(self.config.get('global', 'packages_blacklist'), allow_none=True)
if self.config.has_option('global', 'host_profiles'):
self._host_profiles = ensure_list(self.config.get('global', 'host_profiles'), allow_none=True)
if self.config.has_option('global', 'locales'):
self.locales = ensure_list(self.config.get('global', 'locales'), allow_none=True)
if self.config.has_option('global', 'maturities'):
self.maturities = ensure_list(self.config.get('global', 'maturities'), allow_none=True)
if not self.maturities:
self.maturities = ['PROD']
if self.config.has_option('global', 'default_maturity'):
self.default_maturity = self.config.get('global', 'default_maturity')
if self.config.has_option('global', 'token_lifetime'):
self.token_lifetime = self.config.getint('global', 'token_lifetime')
if self.config.has_option('global', 'use_hostpackages'):
self.use_hostpackages = self.config.getboolean('global', 'use_hostpackages')
if self.config.has_option('global', 'use_ad_groups'):
self.use_ad_groups = self.config.getboolean('global', 'use_ad_groups')
if self.config.has_option('global', 'peercache_enable'):
self.peercache_enable = self.config.getboolean('global', 'peercache_enable')
if self.config.has_option('global', 'waptaudit_task_period'):
self.waptaudit_task_period = self.config.get('global', 'waptaudit_task_period')
self.waptwua_enabled = None
if self.config.has_section('waptwua'):
if self.config.has_option('waptwua', 'enabled'):
self.waptwua_enabled = self.config.getboolean('waptwua', 'enabled')
self.use_repo_rules = False
if self.config.has_option('global', 'use_repo_rules'):
self.use_repo_rules = self.config.getboolean('global', 'use_repo_rules')
self.host_ad_site = None
if self.config.has_option('global', 'host_ad_site'):
self.host_ad_site = self.config.get('global', 'host_ad_site')
self.editor_for_packages = None
if self.config.has_option('global', 'editor_for_packages'):
self.editor_for_packages = self.config.get('global', 'editor_for_packages')
self.limit_bandwidth = None
if self.config.has_option('global', 'limit_bandwidth'):
self.limit_bandwidth = self.config.getfloat('global', 'limit_bandwidth')
if self.config.has_option('global', 'redirect_stdout_to'):
self.redirect_stdout_to = self.config.get('global', 'redirect_stdout_to')
if self.config.has_option('global', 'custom_tags'):
self.custom_tags = ensure_list(self.config.get('global', 'custom_tags'), allow_none=True)
else:
self.custom_tags = []
# clear host key cache
self._host_key = None
# clear host filter for packages
self._packages_filter_for_host = None
# keep the timestamp of last read config files to reload it if it is changed
self.loaded_config_hash = self.get_config_hash()
# backup in DB the merged hash
if merge_config_packages and self.configs_dir and self.merged_config_hash != self.loaded_config_hash:
self.merged_config_hash = self.loaded_config_hash
return self
[docs] def server_uuid(self):
res = self.read_param('server_uuid', public=True)
if not res and self.waptserver:
res = self.waptserver.get('ping')['result']['uuid']
if res:
self.write_param('server_uuid', res, public=True)
return res
[docs] def is_enterprise(self):
try:
# get from database
licences = self.read_param('licences', public=True)
if licences is None:
licences = self.update_licences()
if licences:
if not waptlicences:
return False
waptlicences.check_valid_licences_count(jsondump(licences),self.server_uuid(),36000)
return waptlicences.is_enterprise()
else:
return False
except Exception as e:
logger.warning('Unable to get licence status: %s' % e)
return False
@property
def waptserver(self):
if self._waptserver is None and self.config.has_option('global', 'wapt_server'):
self._waptserver = WaptServer().load_config(self.config)
self._waptserver.capture_external_ip_callback = self.save_external_ip
self.set_client_cert_auth(self._waptserver)
return self._waptserver
@property
def repositories(self)->List[WaptRemoteRepo]:
if self._repositories is None:
# Get the configuration of all repositories (url, ...)
# TODO : make this lazzy...
self._repositories = []
# secondary
if self.config.has_section('global') and self.config.has_option('global', 'repositories'):
repository_names = ensure_list(self.config.get('global', 'repositories'))
logger.info('Other repositories : %s' % (repository_names,))
for name in repository_names:
if name:
w = WaptRepo(name=name, WAPT=self, config=self.config, section=name, cabundle = self.get_cabundle) # we put a callable for cabundle
self.set_client_cert_auth(w)
self._repositories.append(w)
logger.info(' %s:%s' % (w.name, w._repo_url))
else:
repository_names = []
# last is main repository so it overrides the secondary repositories
if self.config.has_option('global', 'repo_url') and not 'wapt' in repository_names:
w = WaptRepo(name='wapt', WAPT=self, config=self.config, cabundle = self.get_cabundle) # we put a callable for cabundle
self._repositories.append(w)
self.set_client_cert_auth(w)
logger.info('Main repository: %s' % (w.repo_url,))
if self.use_hostpackages:
self.add_hosts_repo()
return self._repositories
[docs] def waptwua(self, as_context_manager: bool = False):
def update_server_status(force: bool) -> None:
self.update_server_status(force=force)
class WuaContextManager:
def __init__(self, module):
self.module = module
def __enter__(self):
self.module.enter_wuaserv_ctx()
return self.module
def __exit__(self, exc_type, exc_value, traceback):
self.module.leave_wuaserv_ctx()
self.module.free_instance()
wua = pywaptwua
wua.set_update_server_status_callback(update_server_status)
wua.update_from_wapt(self)
wua.set_certificates_path(self.get_host_certificate_filename(), self.get_host_key_filename())
if as_context_manager:
return WuaContextManager(wua)
else:
return wua
@property
def wua_repository(self):
if self._wua_repository is None:
for r in self.repositories:
if r.name == 'waptwua':
self._wua_repository = r
break
if self._wua_repository is None:
self._wua_repository = WaptRepo(name='waptwua', WAPT=self, config=self.config)
# Fix tls client auth
self.set_client_cert_auth(self._wua_repository)
# Fix verify_cert
if self._wua_repository.verify_cert in [None,1,'1',True] and self.config.has_option('global','verify_cert'):
self._wua_repository.verify_cert = get_verify_cert(self.config.get('global','verify_cert'))
# Fix proxies
if not self._wua_repository.http_proxy and self.config.get('global','use_http_proxy_for_repo') and self.config.get('global','http_proxy') != '':
self._wua_repository.http_proxy = self.config.get('global','http_proxy')
logger.info('WAPTWUA repository: %s' % (self._wua_repository.repo_url,))
return self._wua_repository
[docs] def write_config(self, config_filename=None):
"""Update configuration parameters to supplied inifilename
"""
def _encode_ini_value(value, key=None):
if isinstance(value, list):
return ','.join(value)
elif value is None:
return ''
elif isinstance(value,bool):
if value:
return '1'
else:
return '0'
else:
return str(value)
for key in self.config.defaults():
if hasattr(self, key) and _encode_ini_value(getattr(self, key)) != _encode_ini_value(self.config.defaults()[key]):
logger.debug('update config global.%s : %s' % (key, getattr(self, key)))
self.config.set('global', key, _encode_ini_value(getattr(self, key), key))
repositories_names = ','.join([r.name for r in self.repositories if r.name not in ('global', 'wapt-host')])
if self.config.has_option('global', 'repositories') and repositories_names != '':
self.config.set('global', 'repositories', _encode_ini_value(repositories_names))
if config_filename is None:
config_filename = self.config_filename
if config_filename is not None:
with open(config_filename, 'w', encoding='utf8') as f:
self.config.write(f)
self.loaded_config_hash = self.get_config_hash()
def _set_fake_hostname(self, fqdn):
if sys.platform == 'win32':
import setuphelpers_windows
setuphelpers_windows._fake_hostname = fqdn
else:
setuphelpers._fake_hostname = fqdn
logger.warning('Using test fake hostname and uuid: %s' % fqdn)
self.use_fqdn_as_uuid = True
logger.debug('Host uuid is now: %s' % self.host_uuid)
logger.debug('Host computer_name is now: %s' % setuphelpers.get_computername())
[docs] def get_token_secret_key(self) -> str:
kfn = os.path.join(self.private_dir, 'token_secret_key')
if not os.path.isfile(kfn):
if not os.path.isdir(self.private_dir):
os.makedirs(self.private_dir)
result = ''.join(random.SystemRandom().choice(string.ascii_letters + string.digits) for _ in range(64))
with open(kfn, 'w') as f:
f.write(result)
return result
else:
with open(kfn, 'r') as f:
return f.read()
[docs] def add_hosts_repo(self) -> WaptHostRepo:
"""Add an automatic host repository, remove existing WaptHostRepo last one before"""
# avoid calling getter as the getter is calling this method.
while self._repositories and isinstance(self._repositories[-1], WaptHostRepo):
del self._repositories[-1]
if self.config.has_section('wapt-host'):
section = 'wapt-host'
else:
section = None
if (self.waptserver and self.waptserver.server_url) or section:
try:
# don't create key if not exist at this step
host_key = self.get_host_key(False)
except Exception as e:
logger.debug('Unable to get or create host key: %s' % e)
# unable to access or create host key
host_key = None
host_repo = WaptHostRepo(name='wapt-host', config=self.config, host_id=self.host_packagename(), host_key=host_key, WAPT=self, cabundle = self.get_cabundle) # we put a callable for cabundle
self._repositories.append(host_repo)
# in case host repo is calculated from server url (no specific section) and main repor_url is set
if section is None and self.waptserver and self.waptserver.server_url:
host_repo.repo_url = self.waptserver.server_url+'/wapt-host'
self.set_client_cert_auth(host_repo)
else:
host_repo = None
return host_repo
[docs] def reload_config_if_updated(self):
"""Check if config file has been updated,
Return None if config has not changed or date of new config file if reloaded
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> wapt.reload_config_if_updated()
"""
new_config_hash = self.get_config_hash()
if new_config_hash != self.loaded_config_hash:
tasks_logger.info('Reloading waptcore configuration for Wapt instance thread %s' % threading.get_ident())
self.load_config(merge_config_packages = new_config_hash != self.merged_config_hash)
return True
else:
return False
@property
def waptdb(self) -> WaptDB:
"""Wapt database"""
if not self._waptdb:
self._waptdb = WaptDB(dbpath=self.dbpath)
if not self._waptdb.db_version or self._waptdb.db_version < self._waptdb.curr_db_version:
logger.info('Upgrading db structure from %s to %s' % (self._waptdb.db_version, self._waptdb.curr_db_version))
self._waptdb.upgradedb()
# for backward compat
self._waptdb.execute("""update wapt_localstatus set status_revision=(select cast(value as integer) from wapt_params where name='status_revision') where status_revision is null""")
return self._waptdb
def _migrate_publicdb_data(self):
with self._waptpublicdb:
# duplicate some params from private db to public db
for name in self.public_params:
self._waptpublicdb.set_param(name, self.waptdb.get_param(name))
# migrate install_status
known = ','.join(["'%s'" % p['package_uuid'] for p in self.waptpublicdb.query('select distinct package_uuid from wapt_publicstatus')])
for p in self.waptdb.query("select package_uuid, package, version, architecture, setuppy, install_date from wapt_localstatus where not package_uuid in (%s) and install_status='OK'" % known):
try:
self.waptpublicdb.execute("""\
insert into wapt_publicstatus(package_uuid,package,version,architecture,removal_date,control,setuppy,install_date,uninstall_date)
values (?,?,?,?,?,?,?,?,?)
""", (
p['package_uuid'],
p['package'],
p['version'],
p['architecture'],
None, #removal_date,
PackageEntry().load_control_from_dict(p).ascontrol(),
p['setuppy'],
p['install_date'], #install_date,
None, #uninstall_date
)
)
except Exception as e:
logger.critical('error migrating installed package %s: %s' % (p['package'],e))
@property
def waptpublicdb(self) -> WaptPublicDB:
"""Wapt database"""
if not self._waptpublicdb:
must_migrate_data = not os.path.exists(self.publicdbpath)
self._waptpublicdb = WaptPublicDB(dbpath=self.publicdbpath)
if self._waptpublicdb.db_version < self._waptpublicdb.curr_db_version:
with self._waptpublicdb:
logger.info('Upgrading public db structure from %s to %s' % (self._waptpublicdb.db_version, self._waptpublicdb.curr_db_version))
self._waptpublicdb.upgradedb()
if must_migrate_data:
self._migrate_publicdb_data()
return self._waptpublicdb
@property
def waptsessiondb(self) -> WaptSessionDB:
"""Wapt user session database"""
if not self._waptsessiondb:
self._waptsessiondb = WaptSessionDB(username=setuphelpers.get_current_user())
if self._waptsessiondb.db_version < self._waptsessiondb.curr_db_version:
logger.info('Upgrading db structure from %s to %s' % (self._waptsessiondb.db_version, self._waptsessiondb.curr_db_version))
self._waptsessiondb.upgradedb()
return self._waptsessiondb
@property
def runstatus(self) -> str:
"""returns the current run status for tray display"""
return self.read_param('runstatus', '')
@runstatus.setter
def runstatus(self, waptstatus):
"""Stores in local db the current run status for tray display"""
if self.runstatus != waptstatus:
logger.info('Status : %s' % ensure_unicode(waptstatus))
self.write_param('runstatus', waptstatus)
[docs] def get_hardware_uuid(self):
"""Return '' if no hardware uuid can be found"""
new_hardware_uuid = ''
try:
if waptlicences:
new_hardware_uuid = waptlicences.get_bios_infos()['uuid']
new_hardware_uuid = new_hardware_uuid.lower()
except:
pass
if not new_hardware_uuid:
try:
if os.name == 'nt':
inv = setuphelpers.wmi_info_basic()
new_hardware_uuid = inv.get('System_Information',[{}])[0].get('UUID','').lower()
if not new_hardware_uuid:
inv = setuphelpers.dmi_info()
new_hardware_uuid = inv.get('System_Information',{}).get('UUID','').lower()
else:
inv = setuphelpers.dmi_info()
new_hardware_uuid = inv.get('System_Information',{}).get('UUID','').lower()
if not new_hardware_uuid or new_hardware_uuid in [u.lower() for u in bad_uuid] or (new_hardware_uuid and ' ' in new_hardware_uuid):
logger.info('UUID is a bad uuid' % new_hardware_uuid)
new_hardware_uuid = ''
except:
new_hardware_uuid = ''
return new_hardware_uuid
@property
def host_uuid(self) -> str:
previous_uuid = self.read_param('uuid', public=True) or None
# we ignore changes of char case
if self._host_uuid is None or previous_uuid is None or self._host_uuid.lower() != previous_uuid.lower():
# hostname and hardware uuid last time uuid was set
registered_hostname = self.read_param('hostname', public=True)
registered_hardware_uuid = self.read_param('hardware_uuid',None, public=True)
current_hostname = setuphelpers.get_hostname()
new_hardware_uuid = self.get_hardware_uuid()
if not new_hardware_uuid is None and registered_hardware_uuid is not None:
new_hardware_uuid = registered_hardware_uuid
# track changes case insensitive to handle old data
if (
registered_hostname is None or registered_hostname.lower() != current_hostname.lower() or
registered_hardware_uuid is None or registered_hardware_uuid.lower() != new_hardware_uuid.lower() or
(self.forced_uuid is not None and self.forced_uuid != previous_uuid) or
(self.use_fqdn_as_uuid and previous_uuid.lower() != current_hostname.lower()) or
previous_uuid is None
):
try:
new_uuid = None
# calc the new uuid
# we take in account forced uuid
# forced uuid is for testing.
if self.forced_uuid:
new_uuid = self.forced_uuid.lower()
elif self.use_fqdn_as_uuid:
new_uuid = current_hostname.lower()
# we detect current host changes in case wapt has been cloned on another hardware.
if not new_uuid:
new_uuid = new_hardware_uuid
# preserve initial case to preserve backward compatibility
if previous_uuid and new_uuid and previous_uuid.lower() == new_uuid.lower():
new_uuid = previous_uuid
if not new_uuid:
new_uuid = self.generate_host_uuid()
with self.waptdb, self.waptpublicdb:
# for backward compatibility
self.write_param('uuid', new_uuid, public=True)
self.write_param('hostname', current_hostname, public=True)
self.write_param('hardware_uuid', new_hardware_uuid, public=True)
except:
# no write access
pass
self._host_uuid = new_uuid
else:
self._host_uuid = previous_uuid
return self._host_uuid
@host_uuid.setter
def host_uuid(self, value):
self._host_uuid= None
with self.waptdb, self.waptpublicdb:
self.write_param('uuid', value, public=True)
self.write_param('hostname', setuphelpers.get_hostname(), public=True)
self.write_param('hardware_uuid', self.get_hardware_uuid(), public=True)
@host_uuid.deleter
def host_uuid(self):
self._host_uuid= None
self.forced_uuid = None
self.delete_param('uuid')
[docs] def generate_host_uuid(self):
"""Regenerate a random UUID for this host or force with supplied one.
Normally, the UUID is taken from BIOS through wmi.
In case bios returns some duplicates or garbage, it can be useful to
force a random uuid. This is stored as uuid key in wapt-get.ini.
In case we want to link th host with a an existing record on server, we
can force a old UUID.
"""
auuid = ('rnd-%s' % str(uuid.uuid4())).lower()
self.host_uuid = auuid
return auuid
[docs] def reset_host_uuid(self,new_uuid=None):
"""Reset host uuid to bios provided UUID.
If it was forced in ini file, remove setting from ini file.
"""
with self.waptdb,self.waptpublicdb:
self.delete_param('uuid')
self.delete_param('hardware_uuid')
self.delete_param('hostname')
self._host_uuid = None
self.forced_uuid = None
try:
ini = RawConfigParser()
ini.read(self.config_filename)
if ini.has_option('global', 'uuid') or ini.has_option('default_global', 'uuid'):
if ini.has_option('global', 'uuid'):
ini.remove_option('global', 'uuid')
if ini.has_option('default_global', 'uuid'):
ini.remove_option('default_global', 'uuid')
with open(self.config_filename, 'w') as f:
ini.write(f)
except:
pass
if not new_uuid is None:
self.host_uuid = new_uuid
return self.host_uuid
@property
def host_organizational_unit_dn(self):
"""Get host org unit DN from wapt-get.ini [global] host_organizational_unit_dn if defined
or from registry as supplied by AD / GPO process
"""
host_organizational_unit_dn = self.read_param('host_organizational_unit_dn', None, public=True)
if host_organizational_unit_dn:
return host_organizational_unit_dn
if sys.platform.startswith('linux') or sys.platform.startswith('darwin'):
gpo_host_dn = self.get_cache_domain_info()['ou']
else:
gpo_host_dn = setuphelpers.registry_get(HKEY_LOCAL_MACHINE, r'SOFTWARE\Microsoft\Windows\CurrentVersion\Group Policy\State\Machine', 'Distinguished-Name')
if gpo_host_dn:
try:
default_organizational_unit_dn = ','.join(gpo_host_dn.split(',')[1:])
except:
default_organizational_unit_dn = None
else:
default_organizational_unit_dn = None
return default_organizational_unit_dn
@host_organizational_unit_dn.setter
def host_organizational_unit_dn(self, value):
self.write_param('host_organizational_unit_dn', value, public=True)
@host_organizational_unit_dn.deleter
def host_organizational_unit_dn(self):
self.delete_param('host_organizational_unit_dn', public=True)
[docs] def reset_host_organizational_unit_dn(self):
"""Reset forced host_organizational_unit_dn to AD / GPO registry defaults.
If it was forced in ini file, remove setting from ini file.
"""
del(self.host_organizational_unit_dn)
ini = RawConfigParser()
ini.read(self.config_filename)
if ini.has_option('global', 'host_organizational_unit_dn'):
ini.remove_option('global', 'host_organizational_unit_dn')
with open(self.config_filename, 'w') as f:
ini.write(f)
return self.host_dn
@property
def host_dn(self):
result = 'CN=%s' % setuphelpers.get_computername().upper()
org_unit = self.host_organizational_unit_dn
if org_unit:
result = result + ',' + org_unit
return result
@property
def host_site(self):
return self.get_host_site()
[docs] def check_install_running(self, max_ttl=60):
""" Check if an install is in progress, return list of pids of install in progress
Kill old stucked wapt-get processes/children and update db status
max_ttl is maximum age of wapt-get in minutes
"""
logger.debug('Checking if old install in progress')
# kill old wapt-get
mindate = time.time() - max_ttl*60
killed = []
for p in psutil.process_iter():
try:
if p.pid != os.getpid() and (p.create_time() < mindate) and p.name() in ('wapt-get', 'wapt-get.exe'):
logger.debug('Killing process tree of pid %i' % p.pid)
killtree(p.pid)
logger.debug('Killing pid %i' % p.pid)
killed.append(p.pid)
except (psutil.NoSuchProcess, psutil.AccessDenied):
pass
# reset install_status
logger.debug('reset stalled install_status in database')
init_run_pids = self.waptdb.query("""\
select process_id from wapt_localstatus
where install_status in ('INIT','RUNNING')
""")
all_pids = psutil.pids()
reset_error = []
result = []
for rec in init_run_pids:
# check if process is no more running
if not rec['process_id'] in all_pids or rec['process_id'] in killed:
reset_error.append(rec['process_id'])
else:
# install in progress
result.append(rec['process_id'])
if reset_error:
with self.waptdb:
self.waptdb.execute("""\
update wapt_localstatus
set install_status=coalesce('ERROR',install_status) where process_id in (?)
""", (','.join([str(p) for p in reset_error]),))
self.runstatus = ''
# return pids of install in progress
return result
@property
def pre_shutdown_timeout(self):
"""get / set the pre shutdown timeout shutdown tasks.
"""
if setuphelpers.reg_key_exists(HKEY_LOCAL_MACHINE, r'SYSTEM\CurrentControlSet\services\gpsvc'):
with setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, r'SYSTEM\CurrentControlSet\services\gpsvc') as key:
ms = int(setuphelpers.reg_getvalue(key, 'PreshutdownTimeout', 0))
if ms:
return ms / (60*1000)
else:
return None
else:
return None
@pre_shutdown_timeout.setter
def pre_shutdown_timeout(self, minutes):
"""Set PreshutdownTimeout"""
if setuphelpers.reg_key_exists(HKEY_LOCAL_MACHINE, r'SYSTEM\CurrentControlSet\services\gpsvc'):
key = setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, r'SYSTEM\CurrentControlSet\services\gpsvc', sam=setuphelpers.KEY_WRITE)
if not key:
raise Exception('The PreshutdownTimeout can only be changed with System Account rights')
setuphelpers.reg_setvalue(key, 'PreshutdownTimeout', minutes*60*1000, setuphelpers.REG_DWORD)
@property
def max_gpo_script_wait(self):
"""get / set the MaxGPOScriptWait.
"""
with setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, r'SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System') as key:
ms = int(setuphelpers.reg_getvalue(key, 'MaxGPOScriptWait', 0))
if ms:
return ms / (60*1000)
else:
return None
@max_gpo_script_wait.setter
def max_gpo_script_wait(self, minutes):
"""Set MaxGPOScriptWait"""
key = setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, r'SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System', sam=setuphelpers.KEY_WRITE)
if not key:
raise Exception('The MaxGPOScriptWait can only be changed with System Account rights')
setuphelpers.reg_setvalue(key, 'MaxGPOScriptWait', minutes*60*1000, setuphelpers.REG_DWORD)
@property
def hiberboot_enabled(self):
"""get HiberbootEnabled.
"""
key = setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, r'SYSTEM\CurrentControlSet\Control\Session Manager\Power')
if not key:
return None
try:
return winreg.QueryValueEx(key, 'HiberbootEnabled')[0]
except:
return None
@hiberboot_enabled.setter
def hiberboot_enabled(self, enabled):
"""Set HiberbootEnabled (0/1)"""
key = setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, 'SYSTEM\CurrentControlSet\Control\Session Manager\Power', sam=setuphelpers.KEY_WRITE)
if key:
setuphelpers.reg_setvalue(key, 'HiberbootEnabled', 1 if enabled else 0, setuphelpers.REG_DWORD)
[docs] def registry_uninstall_snapshot(self):
"""Return list of uninstall ID from registry
launched before and after an installation to capture uninstallkey
"""
result = []
with setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, "Software\\Microsoft\\Windows\\CurrentVersion\\Uninstall") as key:
try:
i = 0
while True:
subkey = EnumKey(key, i)
result.append(subkey)
i += 1
except WindowsError as e:
# WindowsError: [Errno 259] No more data is available
if e.winerror == 259:
pass
else:
raise
if platform.machine() == 'AMD64':
with setuphelpers.reg_openkey_noredir(HKEY_LOCAL_MACHINE, "Software\\Wow6432Node\\Microsoft\\Windows\\CurrentVersion\\Uninstall") as key:
try:
i = 0
while True:
subkey = EnumKey(key, i)
result.append(subkey)
i += 1
except WindowsError as e:
# WindowsError: [Errno 259] No more data is available
if e.winerror == 259:
pass
else:
raise
return result
[docs] def uninstall_cmd(self, guid):
"""return the (quiet) command stored in registry to uninstall a software given its registry key"""
return setuphelpers.uninstall_cmd(guid)
[docs] def check_cancelled(self, msg='Task cancelled'):
if self.task_is_cancelled.is_set():
raise EWaptCancelled(msg)
[docs] def run(self, *arg, **args):
if platform.system() == 'Windows':
return ensure_unicode(run(*arg, pidlist=self.pidlist,**args))
return ensure_unicode(run(*arg, **args))
[docs] def run_notfatal(self, *cmd, **args):
"""Runs the command and wait for it termination
returns output, don't raise exception if exitcode is not null but return '' """
try:
return self.run(*cmd, accept_returncodes=None, **args)
except Exception as e:
return ensure_unicode(e)
[docs] def update_public_packages_availability(self):
"""Update package removal_date on publicdb"""
now = datetime2isodate()
with self.waptpublicdb:
"""package_uuid,package,version,architecture,removal_date,control,setuppy,install_date,uninstall_date"""
try:
self.waptpublicdb.db.execute('create temp table temp_package_uuids (package_uuid varchar)')
self.waptpublicdb.db.executemany("""insert into temp_package_uuids(package_uuid) values (?)""", self.waptdb.query('select package_uuid from wapt_package', as_dict=False))
self.waptpublicdb.db.execute("""update wapt_publicstatus set removal_date=? where package_uuid not in (select package_uuid from temp_package_uuids) and removal_date is null""", [now])
finally:
self.waptpublicdb.db.execute('drop table temp_package_uuids')
[docs] def store_public_setuppy_control(self, package_entry:PackageEntry, install_date:str=None):
"""Update status of package installation on publicdb"""
setupfn = os.path.join(package_entry.sourcespath,'setup.py')
if os.path.isfile(setupfn):
setuppy = remove_encoding_declaration(codecs.open(setupfn, 'r', encoding='utf-8').read())
else:
setuppy = ''
(control, fname) = package_entry._control_str()
now = datetime2isodate()
with self.waptpublicdb:
"""package_uuid,package,version,architecture,removal_date,control,setuppy,install_date,uninstall_date"""
cur = self.waptpublicdb.execute("""delete from wapt_publicstatus where package_uuid=?""",(package_entry.package_uuid,))
self.waptpublicdb.execute("""\
insert into wapt_publicstatus(package_uuid,package,version,architecture,removal_date,control,setuppy,install_date,uninstall_date)
values (?,?,?,?,?,?,?,?,?)
""", (
package_entry.package_uuid,
package_entry.package,
package_entry.version,
package_entry.architecture,
None, #removal_date,
control,
setuppy,
install_date or now, #install_date,
None, #uninstall_date
)
)
try:
self.waptpublicdb.db.execute('create temp table temp_package_uuids (package_uuid varchar primary key)')
self.waptpublicdb.db.executemany("""insert into temp_package_uuids(package_uuid) values (?)""",
self.waptdb.query("""select distinct package_uuid from wapt_localstatus where install_status='OK'""", as_dict=False))
self.waptpublicdb.db.execute("""update wapt_publicstatus set uninstall_date=? where package_uuid <> ? and uninstall_date is NULL and package_uuid not in (select package_uuid from temp_package_uuids)""", [now,package_entry.package_uuid])
finally:
self.waptpublicdb.db.execute('drop table temp_package_uuids')
# invalidate all other installations of this package
self.waptpublicdb.execute("""update wapt_publicstatus set uninstall_date=? where uninstall_date is NULL and package=? and package_uuid <> ?""",[now, package_entry.package, package_entry.package_uuid])
[docs] def remove_package_persistent_dirs(self, package_uuid):
if package_uuid and not is_unsafe_filename(package_uuid):
old_persistent_dir = os.path.join(self.persistent_root_dir, package_uuid)
if os.path.isdir(old_persistent_dir):
logger.info('Cleanup of previous versions of {%s} persistent dir: %s' % (package_uuid, old_persistent_dir))
shutil.rmtree(old_persistent_dir)
old_public_persistent_dir = os.path.join(self.public_persistent_root_dir, package_uuid)
if os.path.isdir(old_public_persistent_dir):
logger.info('Cleanup of previous versions of {%s} public persistent dir: %s' % (package_uuid, old_public_persistent_dir))
shutil.rmtree(old_public_persistent_dir)
[docs] def install_wapt(self, fname, params_dict={}, explicit_by=None, force=None):
"""Install a single wapt package given its WAPT filename.
return install status
Args:
fname (str): Path to wapt Zip file or unzipped development directory
params (dict): custom parameters for the install function
explicit_by (str): identify who has initiated the install
Returns:
str: 'OK','ERROR'
Raises:
EWaptMissingCertificate
EWaptNeedsNewerAgent
EWaptUnavailablePackage
EWaptConflictingPackage
EWaptBadPackageAttribute
EWaptException
various Exception depending on setup script
"""
install_id = None
# we record old sys.path as we will include current setup.py
oldpath = sys.path
self.check_cancelled('Install of %s cancelled before starting up' % ensure_unicode(fname))
logger.info("Register start of install %s as user %s to local DB with params %s" % (ensure_unicode(fname), setuphelpers.get_current_user(), params_dict))
logger.info("Interactive user:%s, usergroups %s" % (self.user, self.usergroups))
#if sys.platform == 'win32':
# previous_uninstall = self.registry_uninstall_snapshot()
try:
if not self.cabundle:
raise EWaptMissingCertificate('install_wapt %s: No public Key provided for package signature checking.' % (fname,))
entry = PackageEntry(waptfile=fname)
if not entry.package_uuid:
entry.make_uuid()
logger.info('No uuid, generating package uuid on the fly: %s' % entry.package_uuid)
self.runstatus = "Installing package %s version %s ..." % (entry.package, entry.version)
params = self.get_previous_package_params(entry)
params.update(params_dict)
install_id = self.waptdb.add_start_install(
entry,
params_dict=params,
explicit_by=explicit_by,
)
# we setup a redirection of stdout to catch print output from install scripts
with WaptPackageInstallLogger(sys.stderr, wapt_context=self, install_id=install_id, user=self.user, exit_status=None) as dblogger:
if entry.min_wapt_version and Version(entry.min_wapt_version) > Version(__version__):
raise EWaptNeedsNewerAgent('This package requires a newer Wapt agent. Minimum version: %s' % entry.min_wapt_version)
depends = ensure_list(entry.depends)
conflicts = ensure_list(entry.conflicts)
missing_depends = [p for p in depends if not self.is_installed(p)]
installed_conflicts = [p for p in conflicts if self.is_installed(p)]
if missing_depends:
raise EWaptUnavailablePackage('Missing dependencies: %s' % (','.join(missing_depends,)))
if installed_conflicts:
raise EWaptConflictingPackage('Conflicting packages installed: %s' % (','.join(installed_conflicts,)))
free_disk_space = setuphelpers.get_disk_free_space(os.path.abspath(os.path.join(self.config_filename, os.pardir)))
if entry.installed_size and free_disk_space < entry.installed_size :
raise EWaptDiskSpace('This package requires at least %s free space. Your drive where WAPT is installed has only %s free space' %
(format_bytes(entry.installed_size), format_bytes(free_disk_space)))
entry.check_package_attributes()
errors = []
if not self.host_capabilities().is_matching_package(entry, errors_list=errors):
raise EWaptBadPackageAttribute('This package have an attribute in the control file incompatible with your host capabilities: \n%s' % ",".join(errors))
# don't check in developper mode
if os.path.isfile(fname):
cert = entry.check_control_signature(self.cabundle)
logger.info('Control data for package %s verified by certificate %s' % (setuphelpers.ensure_unicode(fname), cert))
else:
logger.info('Developper mode, don''t check control signature for %s' % setuphelpers.ensure_unicode(fname))
self.check_cancelled()
logger.info("Installing package %s" % (ensure_unicode(fname),))
# case where fname is a wapt zipped file, else directory (during developement)
istemporary = False
if os.path.isfile(fname):
# check signature and files when unzipping
packagetempdir = entry.unzip_package(cabundle=self.cabundle, target_dir = tempfile.mkdtemp(prefix='wapt',dir=self.wapt_temp_dir))
istemporary = True
elif os.path.isdir(fname):
packagetempdir = fname
else:
raise EWaptNotAPackage('%s is not a file nor a directory, aborting.' % ensure_unicode(fname))
try:
previous_cwd = os.getcwd()
self.check_cancelled()
exitstatus = None
new_uninstall_key = None
uninstallstring = None
persistent_source_dir = os.path.join(packagetempdir, 'WAPT', 'persistent')
public_persistent_source_dir = os.path.join(packagetempdir, 'WAPT', 'public_persistent')
if entry.package_uuid and not is_unsafe_filename(entry.package_uuid):
persistent_dir = os.path.join(self.persistent_root_dir, entry.package_uuid)
if os.path.isdir(persistent_dir):
logger.debug('Removing existing persistent dir %s' % persistent_dir)
shutil.rmtree(persistent_dir, ignore_errors=False)
# install persistent files
if os.path.isdir(persistent_source_dir):
logger.info('Copy persistent package data to %s' % persistent_dir)
shutil.copytree(persistent_source_dir, persistent_dir)
else:
# create always
os.makedirs(persistent_dir)
## public
public_persistent_dir = os.path.join(self.public_persistent_root_dir, entry.package_uuid)
if os.path.isdir(public_persistent_dir):
logger.debug('Removing existing public persistent dir %s' % public_persistent_dir)
shutil.rmtree(public_persistent_dir, ignore_errors=False)
# install persistent files
if os.path.isdir(public_persistent_source_dir):
logger.info('Copy public persistent package data to %s' % public_persistent_dir)
shutil.copytree(public_persistent_source_dir, public_persistent_dir)
else:
# create always
os.makedirs(public_persistent_dir)
else:
persistent_dir = persistent_source_dir
public_persistent_dir = public_persistent_source_dir
setup_filename = os.path.join(packagetempdir, 'setup.py')
# take in account the case we have no setup.py
if os.path.isfile(setup_filename):
os.chdir(os.path.dirname(setup_filename))
if not os.getcwd() in sys.path:
sys.path.append(os.getcwd())
# import the setup module from package file
logger.info(" sourcing install file %s " % ensure_unicode(setup_filename))
setup = import_setup(setup_filename)
required_params = []
# be sure some minimal functions are available in setup module at install step
setattr(setup, 'basedir', os.path.dirname(setup_filename))
# redefine run to add reference to wapt.pidlist
setattr(setup, 'run', self.run)
setattr(setup, 'run_notfatal', self.run_notfatal)
if not hasattr(setup, 'uninstallkey'):
setup.uninstallkey = []
# to set some contextual default arguments
def with_install_context(func, impacted_process=None, uninstallkeylist=None, force=None, pidlist=None):
def new_func(*args, **kwargs):
if impacted_process and not 'killbefore' in kwargs:
kwargs['killbefore'] = ensure_list(impacted_process)
if uninstallkeylist is not None and not 'uninstallkeylist' in kwargs:
kwargs['uninstallkeylist'] = uninstallkeylist
if force is not None and not 'force' in kwargs:
kwargs['force'] = force
if pidlist is not None and not 'pidlist' in kwargs:
kwargs['pidlist'] = pidlist
return func(*args, **kwargs)
return new_func
if sys.platform == 'win32':
setattr(setup, 'install_msi_if_needed', with_install_context(
setuphelpers.install_msi_if_needed,
impacted_process=ensure_list(entry.impacted_process),
uninstallkeylist=setup.uninstallkey,
force=force,
pidlist=self.pidlist))
setattr(setup, 'install_exe_if_needed', with_install_context(
setuphelpers.install_exe_if_needed,
impacted_process=ensure_list(entry.impacted_process),
uninstallkeylist=setup.uninstallkey,
force=force,
pidlist=self.pidlist))
if sys.platform == 'darwin':
setattr(setup, 'install_dmg', with_install_context(setuphelpers.install_dmg,
impacted_process=ensure_list(entry.impacted_process),
uninstallkeylist=setup.uninstallkey,
force=force))
setattr(setup, 'install_pkg', with_install_context(setuphelpers.install_pkg,
impacted_process=ensure_list(entry.impacted_process),
uninstallkeylist=setup.uninstallkey,
force=force))
setattr(setup, 'install_app', with_install_context(setuphelpers.install_app,
impacted_process=ensure_list(entry.impacted_process),
uninstallkeylist=setup.uninstallkey,
force=force))
setattr(setup, 'WAPT', self)
setattr(setup, 'control', entry)
setattr(setup, 'language', self.language)
setattr(setup, 'force', force)
setattr(setup, 'user', explicit_by or self.user)
setattr(setup, 'usergroups', self.usergroups)
setattr(setup, 'persistent_source_dir', persistent_source_dir)
setattr(setup, 'persistent_dir', persistent_dir)
setattr(setup, 'public_persistent_source_dir', public_persistent_source_dir)
setattr(setup, 'public_persistent_dir', public_persistent_dir)
# get definitions of required parameters from setup module
if hasattr(setup, 'required_params'):
required_params = setup.required_params
# get value of required parameters if not already supplied
for p in required_params:
if not p in params:
if not is_system_user():
params[p] = input("%s: " % p)
else:
raise EWaptException('Required parameters %s is not supplied' % p)
logger.info('Install parameters : %s' % (params,))
# set params dictionary
if not hasattr(setup, 'params'):
# create a params variable for the setup module
setattr(setup, 'params', params)
else:
# update the already created params with additional params from command line
setup.params.update(params)
# store source of install and params in DB for future use (upgrade, session_setup, uninstall)
with self.waptdb:
self.waptdb.execute("""\
update wapt_localstatus
set setuppy=?,install_params=?,control=? where rowid = ?
""", (
remove_encoding_declaration(codecs.open(setup_filename, 'r', encoding='utf-8').read()),
jsondump(params),
entry._control_str()[0],
install_id,
)
)
with _disable_file_system_redirection():
try:
logger.info(" executing install script")
exitstatus = setup.install()
except Exception as e:
logger.critical('Fatal error in install script: %s:\n%s' % (ensure_unicode(e), ensure_unicode(traceback.format_exc())))
raise
if exitstatus is None or exitstatus == 0:
dblogger.exit_status = 'OK'
else:
dblogger.exit_status = exitstatus
if sys.platform == 'win32':
# get uninstallkey from setup module (string or array of strings)
if hasattr(setup, 'uninstallkey'):
new_uninstall_key = ensure_list(setup.uninstallkey)[:]
# check that uninstallkey(s) are in registry
key_errors = []
for key in new_uninstall_key:
if not setuphelpers.uninstall_key_exists(uninstallkey=key):
key_errors.append(key)
if key_errors:
if len(key_errors) > 1:
raise EWaptException('The uninstall keys: \n%s\n have not been found in system registry after softwares installation.' % ('\n'.join(key_errors),))
else:
raise EWaptException('The uninstall key: %s has not been found in system registry after software installation.' % (' '.join(key_errors),))
else:
if sys.platform == 'darwin':
if hasattr(setup, 'uninstallkey'):
new_uninstall_key = ensure_list(setup.uninstallkey)[:]
key_errors = []
for key in new_uninstall_key:
if not setuphelpers.uninstall_key_exists(key):
key_errors.append(key)
if key_errors :
if len(key_errors) > 1:
raise EWaptException('The uninstall keys: \n%s\n have not been found in system after softwares installation.' % ('\n'.join(key_errors),))
else:
raise EWaptException('The uninstall key: %s has not been found in system after software installation.' % (' '.join(key_errors),))
else:
new_uninstall_key = []
# get uninstallstring from setup module (string or array of strings)
if hasattr(setup, 'uninstallstring'):
uninstallstring = setup.uninstallstring[:]
else:
uninstallstring = None
logger.info(' uninstall keys : %s' % (new_uninstall_key,))
logger.info(' uninstall strings : %s' % (uninstallstring,))
logger.info("Install script finished with status %s" % dblogger.exit_status)
else:
logger.info('No setup.py')
dblogger.exit_status = 'OK'
if entry.package_uuid:
for row in self.waptdb.query('select package_uuid from wapt_localstatus l where l.package=? and l.package_uuid<>?', (entry.package, entry.package_uuid)):
self.remove_package_persistent_dirs(row.get('package_uuid'))
self.waptdb.update_install_status(install_id,
uninstall_key=jsondump(new_uninstall_key))
# provide control and setuppy publicly for session_setup
if dblogger.exit_status == 'OK':
self.store_public_setuppy_control(entry)
finally:
if istemporary:
os.chdir(previous_cwd)
logger.debug("Cleaning package tmp dir")
# trying 3 times to remove
cnt = 3
while cnt > 0:
try:
shutil.rmtree(packagetempdir)
break
except Exception as e:
cnt -= 1
time.sleep(2)
logger.warning(e)
else:
logger.error("Unable to clean tmp dir")
# end
return self.waptdb.install_status(install_id)
except Exception as e:
if install_id:
try:
self.waptdb.update_install_status(install_id, set_status='ERROR', append_line=ensure_unicode(e))
except Exception as e2:
logger.critical(ensure_unicode(e2))
else:
logger.critical(ensure_unicode(e))
raise e
finally:
gc.collect()
if 'setup' in dir() and setup is not None:
setup_name = setup.__name__[:]
logger.debug('Removing module: %s, refcnt: %s' % (setup_name, sys.getrefcount(setup)))
del setup
if setup_name in sys.modules:
del sys.modules[setup_name]
sys.path = oldpath
self.store_upgrade_status()
self.runstatus = ''
[docs] def running_tasks(self):
"""return current install tasks"""
running = self.waptdb.query_package_entry("""\
select * from wapt_localstatus
where install_status in ('INIT','DOWNLOAD','RUNNING')
""")
return running
[docs] def error_packages(self):
"""return install tasks with error status"""
q = self.waptdb.query_package_entry("""\
select * from wapt_localstatus
where install_status in ('ERROR')
""")
return q
[docs] def store_upgrade_status(self, upgrades=None):
"""Stores in DB the current pending upgrades and running installs for
query by waptservice"""
try:
status = {
"running_tasks": ["%s : %s" % (p.asrequirement(), p.install_status) for p in self.running_tasks()],
"errors": ["%s" % p.asrequirement() for p in self.error_packages()],
"date": datetime2isodate(),
}
if upgrades is None:
upgrades = self.list_upgrade()
status["upgrades"] = upgrades['upgrade']+upgrades['install']+upgrades['additional']
status["pending"] = upgrades
logger.debug("store status in DB")
self.write_param('last_update_status', status)
return status
except Exception as e:
logger.critical('Unable to store status of update in DB : %s' % ensure_unicode(e))
if logger.level == logging.DEBUG:
raise
[docs] def read_upgrade_status(self):
"""Return last stored pending updates status
Returns:
dict: {running_tasks errors pending (dict) upgrades (list)}
"""
return self.read_param('last_update_status', ptype='json')
[docs] def get_sources(self, package):
"""Download sources of package (if referenced in package as a https svn)
in the current directory
Args:
package (str or PackageRequest): package to get sources for
Returns:
str : checkout directory path
"""
sources_url = None
entry = None
entries = self.waptdb.packages_matching(package)
if entries:
entry = entries[-1]
if entry.sources:
sources_url = entry.sources
if not sources_url:
if self.config.has_option('global', 'default_sources_url'):
sources_url = self.config.get('global', 'default_sources_url') % {'packagename': package}
if not sources_url:
raise Exception('No sources defined in package control file and no default_sources_url in config file')
if "PROGRAMW6432" in os.environ:
svncmd = os.path.join(os.environ['PROGRAMW6432'], 'TortoiseSVN', 'bin', 'svn.exe')
else:
svncmd = os.path.join(os.environ['PROGRAMFILES'], 'TortoiseSVN', 'bin', 'svn.exe')
logger.debug('svn command : %s' % svncmd)
if not os.path.isfile(svncmd):
raise Exception('svn.exe command not available, please install TortoiseSVN with commandline tools')
# checkout directory
if entry:
co_dir = self.get_default_development_dir(entry.package, section=entry.section)
else:
co_dir = self.get_default_development_dir(package)
logger.info('sources : %s' % sources_url)
logger.info('checkout dir : %s' % co_dir)
# if already checked out...
if os.path.isdir(os.path.join(co_dir, '.svn')):
print((self.run('"%s" up "%s"' % (svncmd, co_dir))))
else:
print((self.run('"%s" co "%s" "%s"' % (svncmd, sources_url, co_dir))))
return co_dir
[docs] def last_install_log(self, packagename):
r"""Get the printed output of the last install of package named packagename
Args:
packagename (str): name of package to query
Returns:
dict: {status,log} of the last install of a package
>>> w = Wapt()
>>> w.last_install_log('tis-7zip')
???
{'install_status': u'OK', 'install_output': u'Installing 7-Zip 9.38.0-1\n7-Zip already installed, skipping msi install\n',install_params: ''}
"""
q = self.waptdb.query("""\
select rowid,package,version,architecture,maturity,locale,install_status,
install_output,install_params,explicit_by,uninstall_key,install_date,
last_audit_status,last_audit_on,last_audit_output,next_audit_on,package_uuid
from wapt_localstatus
where package=? order by install_date desc limit 1
""", (packagename,))
if not q:
raise Exception("Package %s not found in local DB status" % packagename)
return q[0]
[docs] def cleanup(self, obsolete_only=False):
"""Remove cached WAPT files from local disk
Args:
obsolete_only (boolean): If True, remove packages which are either no more available,
or installed at a equal or newer version
Returns:
list: list of filenames of removed packages
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> l = wapt.download_packages(wapt.check_downloads())
>>> res = wapt.cleanup(True)
"""
result = []
logger.info('Cleaning up WAPT cache directory')
cachepath = self.packages_cache_dir
upgrade_actions = self.list_upgrade()
futures = upgrade_actions['install'] +\
upgrade_actions['upgrade'] +\
upgrade_actions['additional']
def in_futures(pe):
for p in futures:
if pe.match(p):
return True
return False
for f in glob.glob(os.path.join(cachepath, '*.wapt')):
if os.path.isfile(f):
can_remove = True
if obsolete_only:
try:
# check if cached package could be installed at next ugrade
pe = PackageEntry().load_control_from_wapt(f)
pe_installed = self.is_installed(pe.package)
can_remove = not in_futures(pe) and ((pe_installed and pe <= pe_installed) or not self.is_available(pe.asrequirement()))
except:
# if error... control file in wapt file is corrupted.
continue
if can_remove:
logger.debug('Removing %s' % f)
try:
os.remove(f)
result.append(f)
except Exception as e:
logger.warning('Unable to remove %s : %s' % (f, ensure_unicode(e)))
return result
def _update_db(self, repo, force=False, current_datetime=None):
"""Get Packages from http repo and update local package database
return last-update header
The local status DB is updated. Date of index is stored in params table
for further checks.
Args:
force (bool): get index from remote repo even if creation date is not newer
than the datetime stored in local status database
waptdb (WaptDB): instance of Wapt status database.
current_datetime (str): iso current datetime for package date filters (valid_from, valid_until)
Returns:
isodatetime: date of Packages index
>>> import common
>>> repo = common.WaptRepo('wapt','http://wapt/wapt')
>>> localdb = common.WaptDB('c:/wapt/db/waptdb.sqlite')
>>> last_update = repo.is_available()
>>> repo.update_db(waptdb=localdb) == last_update
True
"""
last_modified = self.waptdb.get_param('last-packages_date-%s' % (repo.name))
# Check if updated
# next_update_is_forced by previous update if there was some valid_from, valid_until, force_install in future
if force or self.waptdb.get_param('next_update_is_forced-%s' % repo.name) or repo.need_update(last_modified):
old_status = repo.invalidate_packages_cache()
discarded = []
self._packages_filter_for_host = None
if self.filter_on_host_cap:
host_capabilities = self.host_capabilities()
else:
host_capabilities = None
with self.waptdb:
try:
logger.info('Read Packages index file for repo %s' % repo.name)
last_modified = repo.packages_date()
if not current_datetime:
current_datetime = datetime2isodate()
self.waptdb.purge_repo(repo.name)
repo_packages = repo.packages()
discarded.extend(repo.discarded)
next_update_on = '9999-12-31'
next_update_is_forced = False
for package in repo_packages:
# if there are time related restriction, we should check again at that time in the future.
if package.valid_from and package.valid_from > current_datetime:
next_update_on = min(next_update_on, package.valid_from)
next_update_is_forced = True
if package.valid_until and package.valid_until > current_datetime:
next_update_on = min(next_update_on, package.valid_until)
next_update_is_forced = True
if package.forced_install_on and package.forced_install_on > current_datetime:
next_update_on = min(next_update_on, package.forced_install_on)
next_update_is_forced = True
if self.filter_on_host_cap:
if not host_capabilities.is_matching_package(package, current_datetime):
discarded.append(package)
continue
try:
self.waptdb.add_package_entry(package, self.language)
except Exception as e:
logger.critical('Error adding entry %s to local DB for repo %s : discarding : %s' % (package.asrequirement(), repo.name, e))
discarded.append(package)
self.waptdb.set_param('last-packages_date-%s' % repo.name, repo.packages_date())
self.waptdb.set_param('next-update-%s' % repo.name, next_update_on)
self.waptdb.set_param('next_update_is_forced-%s' % repo.name, next_update_is_forced)
self.waptdb.set_param('last-discarded-count-%s' % repo.name, len(discarded))
return (last_modified, next_update_on,len(discarded))
except Exception as e:
logger.info('Unable to update repository status of %s, error %s' % (repo._repo_url, e))
# put back cached status data
for (k, v) in old_status.items():
setattr(repo, k, v)
raise
else:
return (self.waptdb.get_param('last-packages_date-%s' % repo.name),
self.waptdb.get_param('next-update-%s' % repo.name, '9999-12-31'),
self.waptdb.get_param('last-discarded-count-%s' % repo.name,0)
)
[docs] def get_host_locales(self):
return ensure_list(self.locales)
[docs] def get_host_site(self):
if self.host_ad_site:
return self.host_ad_site
if sys.platform == 'win32':
return setuphelpers.registry_get(setuphelpers.HKEY_LOCAL_MACHINE, r'SOFTWARE\Microsoft\Windows\CurrentVersion\Group Policy\State\Machine', 'Site-Name')
else:
return self.get_cache_domain_info()['site']
return None
[docs] def get_host_certificate_fingerprint(self):
result = self.read_param('host_certificate_fingerprint')
if result is None:
result = self.get_host_certificate().fingerprint
self.write_param('host_certificate_fingerprint', result)
return result
[docs] def get_host_certificate_authority_key_identifier(self):
result = self.read_param('host_certificate_authority_key_identifier')
if result is None:
result = codecs.encode(self.get_host_certificate().authority_key_identifier or b'', 'hex').decode('ascii')
self.write_param('host_certificate_authority_key_identifier', result)
return result
[docs] def host_capabilities(self):
"""Return the current capabilities of host taken in account to determine packages list and whether update should be forced (when filter criteria are updated)
This includes host certificate,architecture,locale,authorized certificates
Returns:
dict
"""
# be sure to copy and not only reference...
tags = self.custom_tags[:]
if platform.system() == 'Linux':
tags += [(setuphelpers.get_distrib_linux()+'-'+setuphelpers.get_code_name_version()).lower(), setuphelpers.get_distrib_linux().lower(), setuphelpers.get_distrib_linux().lower()+setuphelpers.get_distrib_version().split('.')[0], setuphelpers.get_distrib_linux().lower()+'-'+setuphelpers.get_distrib_version().split('.')[0]]
if setuphelpers.is_debian_based():
tags += ['debian_based']
elif setuphelpers.is_rhel_based():
tags += ['rhel_based', 'redhat_based']
tags += ['linux', 'unix']
os_name = setuphelpers.get_distrib_linux()
elif platform.system() == 'Darwin':
tags += ['macos', 'mac', 'darwin', 'unix']
release_name = setuphelpers.get_release_name()
if release_name:
tags += [release_name]
os_name = "macos"
elif platform.system() == 'Windows':
tags += [('windows' + '-' + platform.release()).lower(), ('win' + '-' + platform.release()).lower(), ('w' + '-' + platform.release()).lower(), ('windows' + platform.release()).lower(), ('win' + platform.release()).lower(), ('w' + platform.release()).lower(), 'windows', 'win', 'w']
os_name = "windows"
return HostCapabilities(
uuid=self.host_uuid,
language=self.language,
os=os_name,
tags=tags,
os_version=setuphelpers.get_os_version(),
kernel_version=setuphelpers.get_kernel_version() if os.name != 'nt' else None,
architecture=setuphelpers.get_host_architecture(),
dn=self.host_dn,
fqdn=setuphelpers.get_hostname(),
site=self.get_host_site(),
wapt_version=Version(__version__, 3),
wapt_edition=self.get_wapt_edition(),
packages_trusted_ca_fingerprints=[c.fingerprint for c in self.authorized_certificates()],
packages_blacklist=self.packages_blacklist,
packages_whitelist=self.packages_whitelist,
packages_locales=self.locales,
packages_maturities=self.maturities,
use_hostpackages=self.use_hostpackages,
host_profiles=self.host_profiles,
host_certificate_fingerprint=self.get_host_certificate_fingerprint(),
host_certificate_authority_key_identifier=self.get_host_certificate_authority_key_identifier(),
host_packages_names=self.get_host_packages_names(),
)
[docs] def packages_filter_for_host(self):
"""Returns a PackageRequest object based on host capabilities to filter applicable packages from a repo
Returns:
PackageRequest
"""
if self._packages_filter_for_host is None:
self._packages_filter_for_host = self.host_capabilities().get_package_request_filter()
return self._packages_filter_for_host
[docs] def get_wapt_edition(self):
return 'enterprise' if self.is_enterprise() else 'discovery'
[docs] def host_capabilities_fingerprint(self):
"""Return a fingerprint representing the current capabilities of host
This includes host certificate,architecture,locale,authorized certificates
Returns:
str
"""
return self.host_capabilities().fingerprint()
[docs] def is_locally_allowed_package(self, package):
"""Return True if package is not in blacklist and is in whitelist if whitelist is not None
packages_whitelist and packages_blacklist are list of package name wildcards (file style wildcards)
blacklist is taken in account first if defined.
whitelist is taken in acoount if not None, else all not blacklisted package names are allowed.
"""
if self.packages_blacklist is not None:
for bl in self.packages_blacklist:
if glob.fnmatch.fnmatch(package.package, bl):
return False
if self.packages_whitelist is None:
return True
else:
for wl in self.packages_whitelist:
if glob.fnmatch.fnmatch(package.package, wl):
return True
return False
def _update_repos_list(self, errors, force=False, current_datetime=None):
"""update the packages database with Packages files from the Wapt repos list
removes obsolete records for repositories which are no more referenced
Args:
force : update repository even if date of packages index is same as
last retrieved date
Returns:
dict: update_db results for each repository name
which has been accessed.
>>> wapt = Wapt(config_filename = 'c:/tranquilit/wapt/tests/wapt-get.ini' )
>>> res = wapt._update_repos_list()
{'wapt': '2018-02-13T11:22:00', 'wapt-host': u'2018-02-09T10:55:04'}
"""
if self.filter_on_host_cap:
# force update if host capabilities have changed and requires a new filering of packages
new_capa = self.host_capabilities_fingerprint()
old_capa = self.read_param('host_capabilities_fingerprint')
if not force and old_capa != new_capa:
logger.info('Host capabilities have changed since last update, forcing update')
force = True
with self.waptdb:
result = {}
logger.debug('Remove unknown repositories from packages table and params (%s)' % (','.join('"%s"' % r.name for r in self.repositories),))
obsolete = self.waptdb.query('select count(*) as cnt from wapt_package where repo not in (%s) or repo is null' % (','.join('"%s"' % r.name for r in self.repositories)))
if obsolete and obsolete[0]['cnt']:
self.waptdb.execute('delete from wapt_package where repo not in (%s) or repo is null' % (','.join('"%s"' % r.name for r in self.repositories)))
obsolete = self.waptdb.query('select count(*) as cnt from wapt_params where name like "last-url-%%" and name not in (%s)' % (','.join('"last-url-%s"' % r.name for r in self.repositories)))
if obsolete and obsolete[0]['cnt']:
self.waptdb.execute('delete from wapt_params where name like "last-url-%%" and name not in (%s)' % (','.join('"last-url-%s"' % r.name for r in self.repositories)))
# to check the next time we should update the local repositories
next_update_on = '9999-12-31'
if not current_datetime:
current_datetime = datetime2isodate()
total_discarded = 0
for repo in self.repositories:
try:
(result[repo.name], repo_next_update_on, discarded) = self._update_db(repo, force=force, current_datetime=current_datetime)
total_discarded += discarded
next_update_on = min(next_update_on, repo_next_update_on)
except Exception as e:
errors.append('Error merging Packages from %s into db: %s' % (repo.name, e))
logger.critical('Error merging Packages from %s into db: %s' % (repo.name, e))
if self.filter_on_host_cap:
self.write_param('host_capabilities_fingerprint', new_capa)
self.write_param('last_update_config_fingerprint', self.merged_config_hash )
self.write_param('next_update_on', next_update_on)
self.waptdb.set_param('last-discarded-count', total_discarded)
return result
[docs] def update_repo_rules(self, force=False):
if self.waptserver:
try:
rules = self.waptserver.get('rules.json', enable_password_callback=False)
new_rules_hash = sha256_for_data(json.dumps(rules, sort_keys=True))
old_rules_hash = self.read_param('repo_rules_sha256-wapt')
if force or (new_rules_hash != old_rules_hash):
rules_verified = []
for rule in rules:
try:
signer_cert_chain = get_cert_chain_from_pem(rule['signer_certificate'])
chain = self.cabundle.check_certificates_chain(signer_cert_chain)
rule['verified_by'] = chain[0].verify_claim(rule, required_attributes=rule['signed_attributes'])
rules_verified.append(rule)
rule['active_rule'] = False
except:
logger.debug('Cert is not trusted or bad signature for : \n%s' % (rule))
self.write_param('repo_rules-wapt', rules_verified)
self.write_param('repo_rules_sha256-wapt', new_rules_hash)
for repo in self._repositories:
repo.reset_network()
self.wua_repository.reset_network()
return True
else:
return False
except:
return False
[docs] def update_licences(self,force=False):
try:
if self.waptserver:
licences = self.waptserver.get('licences.json', enable_password_callback=False)
self.write_param('licences', licences, public=True)
return licences
else:
return None
except:
return None
[docs] def update(self, force=False, register=True):
"""Update local database with packages definition from repositories
Args:
force (boolean): update even if Packages index on repository has not been
updated since last update (based on http headers)
register (boolean): Send informations about status of local packages to waptserver
.. versionadded 1.3.10::
filter_on_host_cap (boolean) : restrict list of retrieved packages to those matching current os / architecture
Returns;
list of (host package entry,entry date on server)
Returns:
dict: {"added","removed","count","repos","upgrades","date"}
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> updates = wapt.update()
>>> 'count' in updates and 'added' in updates and 'upgrades' in updates and 'date' in updates and 'removed' in updates
True
"""
if self.use_repo_rules and self.waptserver:
self.update_repo_rules(force=force)
self.update_licences(force=force)
# check date of wsus
if sys.platform == 'win32' and self.waptwua_enabled:
if self.is_enterprise():
try:
# Set waptwua.status to NEED-SCAN if the server wsuscn2.cab date changed
self.waptwua().check_rescan_needed()
except Exception as e:
logger.debug('Unable to get wsusscn2cab.cab date from server: %s' % e)
previous = self.waptdb.known_packages()
# (main repo is at the end so that it will used in priority)
errors = []
next_update_on = self._update_repos_list(force=force, errors=errors)
current = self.waptdb.known_packages()
result = {
"added": [current[package_uuid] for package_uuid in current if not package_uuid in previous],
"removed": [previous[package_uuid] for package_uuid in previous if not package_uuid in current],
"discarded_count": self.read_param('last-discarded-count'),
"count": len(current),
"repos": [r.name for r in self.repositories],
"upgrades": self.list_upgrade(),
"date": datetime2isodate(),
"next_update_on": next_update_on,
"errors": errors
}
self.update_public_packages_availability()
self.store_upgrade_status(result['upgrades'])
if self.waptserver and not self.disable_update_server_status and register:
try:
self.update_server_status()
except Exception as e:
logger.info('Unable to contact server to register current packages status')
logger.debug('Unable to update server with current status : %s' % e)
if logger.level == logging.DEBUG:
raise
return result
[docs] def update_crls(self, force=False):
# retrieve CRL
# TODO : to be moved to an abstracted wapt https client
crl_dir = setuphelpers.makepath(self.wapt_base_dir, 'ssl', 'crl')
result = []
for cert in self.cabundle.certificates():
crl_urls = cert.crl_urls()
for url in crl_urls:
crl_filename = setuphelpers.makepath(crl_dir, sha256_for_data(str(url))+'.crl')
if os.path.isfile(crl_filename):
ssl_crl = SSLCRL(crl_filename)
else:
ssl_crl = None
if force or not ssl_crl or ssl_crl.next_update > datetime.datetime.utcnow():
try:
# need update
if not os.path.isdir(crl_dir):
os.makedirs(crl_dir)
logger.debug('Download CRL %s' % (url,))
wget(url, target=crl_filename, limit_bandwidth=self.limit_bandwidth)
ssl_crl = SSLCRL(crl_filename)
result.append(ssl_crl)
except Exception as e:
logger.warning('Unable to download CRL from %s: %s' % (url, repr(e)))
if ssl_crl:
result.append(ssl_crl)
pass
elif ssl_crl:
# not changed
result.append(ssl_crl)
return result
[docs] def check_all_depends_conflicts(self):
"""Check the whole dependencies/conflicts tree for installed packages
"""
installed_packages = self.installed(True)
all_depends = defaultdict(list)
all_conflicts = defaultdict(list)
all_missing = defaultdict(list)
if self.use_hostpackages:
for p in self.get_host_packages():
all_depends[p.asrequirement()].append(None)
(depends, conflicts, missing) = self.waptdb.build_depends(p.asrequirement())
for d in depends:
if not p in all_depends[d]:
all_depends[d].append(p.asrequirement())
for c in conflicts:
if not p in all_conflicts[c]:
all_conflicts[c].append(p.asrequirement())
for m in missing:
if not m in all_missing:
all_missing[m].append(p.asrequirement())
for p in installed_packages:
if self.is_locally_allowed_package(p):
if not p.asrequirement() in all_depends:
all_depends[p.asrequirement()] = []
else:
if not p.asrequirement() in all_conflicts:
all_conflicts[p.asrequirement()] = []
(depends, conflicts, missing) = self.waptdb.build_depends(p.asrequirement())
for d in depends:
if not p in all_depends[d]:
all_depends[d].append(p.asrequirement())
for c in conflicts:
if not p in all_conflicts[c]:
all_conflicts[c].append(p.asrequirement())
for m in missing:
if not m in all_missing:
all_missing[m].append(p.asrequirement())
return (all_depends, all_conflicts, all_missing)
[docs] def check_depends(self, apackages, forceupgrade=False, force=False, assume_removed=[], package_request_filter=None):
"""Given a list of packagename or requirement "name (=version)",
return a dictionnary of {'additional' 'upgrade' 'install' 'skipped' 'unavailable','remove'} of
[packagerequest,matching PackageEntry]
Args:
apackages (str or list): list of packages for which to check missing dependencies.
forceupgrade (boolean): if True, check if the current installed packages is the latest available
force (boolean): if True, install the latest version even if the package is already there and match the requirement
assume_removed (list): list of packagename which are assumed to be absent even if they are actually installed to check the
consequences of removal of packages, implies force=True
package_request_filter (PackageRequest): additional filter to apply to packages to sort by locales/arch/mat preferences
if None, get active host filter
Returns:
dict : {'additional' 'upgrade' 'install' 'skipped' 'unavailable', 'remove'} with list of [packagerequest,matching PackageEntry]
"""
if apackages is None:
apackages = []
# additional global scoping
if package_request_filter is None:
package_request_filter = self.packages_filter_for_host()
package_requests = self._ensure_package_requests_list(apackages, package_request_filter=package_request_filter)
if not isinstance(assume_removed, list):
assume_removed = [assume_removed]
if assume_removed:
force = True
# packages to install after skipping already installed ones
skipped = []
unavailable = []
additional_install = []
to_upgrade = []
to_remove = []
packages = []
# search for most recent matching package to install
for package_request in package_requests:
# get the current installed package matching the request
old_matches = self.waptdb.installed_matching(package_request,include_setup=False,include_control=False)
# removes "assumed removed" packages
if old_matches:
for packagename in assume_removed:
if old_matches.match(packagename):
old_matches = None
break
# current installed matches
if not force and old_matches and not forceupgrade:
skipped.append((package_request, old_matches))
else:
new_availables = self.waptdb.packages_matching(package_request)
if new_availables:
if force or not old_matches or (forceupgrade and old_matches < new_availables[-1]):
if not (package_request, new_availables[-1]) in packages:
packages.append((package_request, new_availables[-1]))
else:
skipped.append((package_request, old_matches))
else:
if (package_request, None) not in unavailable:
unavailable.append((package_request, None))
# get dependencies of not installed top packages
if forceupgrade:
(depends, conflicts, missing) = self.waptdb.build_depends(package_requests)
else:
(depends, conflicts, missing) = self.waptdb.build_depends([p[0] for p in packages])
for p in missing:
if (p, None) not in unavailable:
unavailable.append((p, None))
# search for most recent matching package to install
for request in depends:
package_request = PackageRequest(request=request, copy_from=package_request_filter)
# get the current installed package matching the request
old_matches = self.waptdb.installed_matching(package_request,include_setup=False,include_control=False)
# removes "assumed removed" packages
if old_matches:
for packagename in assume_removed:
if old_matches.match(packagename):
old_matches = None
break
# current installed matches
if not force and old_matches:
skipped.append((package_request, old_matches))
else:
# check if installable or upgradable ?
new_availables = self.waptdb.packages_matching(package_request)
if new_availables:
if not old_matches or (forceupgrade and old_matches < new_availables[-1]):
additional_install.append((package_request, new_availables[-1]))
else:
skipped.append((package_request, old_matches))
else:
unavailable.append((package_request, None))
# check new conflicts which should force removal
all_new = additional_install+to_upgrade+packages
def remove_matching(package, req_pe_list):
todel = []
for req, pe in req_pe_list:
if pe.match(package):
todel.append((req, pe))
for e in todel:
req_pe_list.remove(e)
for (request, pe) in all_new:
conflicts = ensure_list(pe.conflicts)
for conflict in conflicts:
installed_conflict = self.waptdb.installed_matching(conflict,include_errors=True, include_setup=False, include_control=False)
if installed_conflict and not ((conflict, installed_conflict)) in to_remove:
to_remove.append((conflict, installed_conflict))
remove_matching(conflict, to_upgrade)
remove_matching(conflict, additional_install)
remove_matching(conflict, skipped)
result = {'additional': additional_install, 'upgrade': to_upgrade, 'install': packages, 'skipped': skipped, 'unavailable': unavailable, 'remove': to_remove}
return result
[docs] def check_remove(self, apackages):
"""Return a list of additional package to remove if apackages are removed
Args:
apackages (str or list of req or PackageRequest): list of packages for which parent dependencies will be checked.
Returns:
list: list of PackageRequest with broken dependencies
"""
if not isinstance(apackages, list):
apackages = [apackages]
result = []
package_requests = self._ensure_package_requests_list(apackages, PackageRequest())
removed_names = []
for r in package_requests:
if not r.package:
packages = self.packages_matching(r)
removed_names.extend([p.package for p in packages])
else:
removed_names.append(r.package)
installed_without_removals = []
for p in self.installed():
match_removals = False
for package_name in removed_names:
if p.package == package_name:
match_removals = True
break
if not match_removals:
installed_without_removals.append(p)
for pe in installed_without_removals:
# test for each installed package if the removal would imply a reinstall
test = self.check_depends(pe, assume_removed=removed_names, package_request_filter=PackageRequest())
# get package names only
reinstall = [p[0] for p in (test['upgrade'] + test['additional'])]
for pr in reinstall:
if pr.package in removed_names and not pe in result:
result.append(pe)
return result
[docs] def remove_is_allowed(self, package: str, usergroups: List[str] )->bool:
# on enterprise, rules are needed to know if packages removal is allowed
rules = None
if self.is_enterprise():
rules = self_service_rules(self)
# check rules for self service
if not self.is_authorized_package_action('remove', package, usergroups, rules):
logger.info('Uninstall of %s not allowed because of selfservice rules.' % (package, ))
return False
additional_removes = self.check_remove(package)
for parent_removed in additional_removes:
if not self.remove_is_allowed(parent_removed.package, usergroups=usergroups):
logger.info('Uninstall of %s not allowed because it is a dependency of a package whose removal is not allowed.' % (package, ))
return False
return True
[docs] def check_install(self, apackages=None, force=True, forceupgrade=True)->Dict[str,List[Tuple]]:
"""Return a list of actions required for install of apackages list of packages
if apackages is None, check for all pending updates.
Args:
apackages (str or list): list of packages or None to check pending install/upgrades
force (boolean): if True, already installed package listed in apackages
will be considered to be reinstalled
forceupgrade: if True, all dependencies are upgraded to latest version,
even if current version comply with depends requirements
Returns:
dict: with keys ['skipped', 'additional', 'remove', 'upgrade', 'install', 'unavailable'] and list of
(package requirements, PackageEntry)
"""
if apackages is None:
actions = self.list_upgrade()
apackages = actions['install']+actions['additional']+actions['upgrade']
actions = self.check_depends(apackages, force=force, forceupgrade=forceupgrade)
return actions
[docs] def packages_matching(self, package_request=None, query=None, args=())->List[PackageEntry]:
"""Returns the list of known packages Entries matching a PackageRequest
Args:
package_request (PackageRequest): request
Returns:
list (of PackageEntry)
"""
if isinstance(package_request, str):
package_request = PackageRequest(request=package_request)
if query is None:
if package_request is not None and package_request.package:
query = 'select * from wapt_package where package=?'
args = (package_request.package,)
else:
query = 'select * from wapt_package'
args = ()
return self.waptdb.query_package_entry(query=query, args=args, package_request=package_request)
def _ensure_package_requests_list(self, package_requests_or_str, package_request_filter=None, keep_package_entries=False)->List[PackageRequest]:
"""Takes a list of packages request as string, or PackageRequest or PackageEntry
and return a list of PackageRequest
Args:
package_requests ( (list of) str,PackageEntry,PackageRequest)
package_request_filter ( PackageRequest) : additional filter. If None, takes the host filter.
Returns:
list of PackageRequest
"""
if package_request_filter is None and self.filter_on_host_cap:
package_request_filter = self.packages_filter_for_host()
package_requests = []
if not isinstance(package_requests_or_str, list):
package_requests_or_str = [package_requests_or_str]
for req in package_requests_or_str:
if isinstance(req, PackageEntry):
if keep_package_entries:
package_requests.append(req)
else:
package_requests.append(PackageRequest(request=req.asrequirement(), copy_from=package_request_filter))
elif isinstance(req, str):
package_requests.append(PackageRequest(request=req, copy_from=package_request_filter))
elif isinstance(req, PackageRequest):
package_requests.append(req)
else:
raise Exception('Unsupported request %s for check_depends' % req)
return package_requests
[docs] def install(self, apackages=List[str],
force=False,
params_dict={},
download_only=False,
usecache=True,
printhook=None,
installed_by=None,
only_priorities=None,
only_if_not_process_running=False,
process_dependencies=True):
"""Install a list of packages and its dependencies
removes first packages which are in conflicts package attribute
Returns a dictionary of (package requirement,package) with 'install','skipped','additional'
Args:
apackages (list or str): list of packages requirements "packagename(=version)" or list of PackageEntry.
force (bool) : reinstalls the packages even if it is already installed
params_dict (dict) : parameters passed to the install() procedure in the packages setup.py of all packages
as params variables and as "setup module" attributes
download_only (bool) : don't install package, but only download them
usecache (bool) : use the already downloaded packages if available in cache directory
printhook (func) : hook for progress print
Returns:
dict: with keys ['skipped', 'additional', 'remove', 'upgrade', 'install', 'unavailable'] and list of
(package requirements, PackageEntry)
>>> wapt = Wapt(config_filename='c:/tranquilit/wapt/tests/wapt-get.ini')
>>> def nullhook(*args):
... pass
>>> res = wapt.install(['tis-wapttest'],usecache=False,printhook=nullhook,params_dict=dict(company='toto'))
>>> isinstance(res['upgrade'],list) and isinstance(res['errors'],list) and isinstance(res['additional'],list) and isinstance(res['install'],list) and isinstance(res['unavailable'],list)
True
>>> res = wapt.remove('tis-wapttest')
>>> res == {'removed': ['tis-wapttest'], 'errors': []}
True
"""
apackages = self._ensure_package_requests_list(apackages, keep_package_entries=True)
# ensure that apackages is a list of package requirements (strings)
logger.info('Trying to install %s with force=%s, only_priorities=%s, only_if_not_process_running=%s' % (repr(apackages), force, only_priorities, only_if_not_process_running))
actions = self.check_depends(apackages = apackages, force=force or download_only, forceupgrade=True)
actions['errors'] = []
packages = actions['install']
skipped = actions['skipped']
not_allowed = []
actions['not_allowed'] = not_allowed
if process_dependencies:
to_upgrade = actions['upgrade']
additional_install = actions['additional']
else:
to_upgrade = []
additional_install = []
# removal from conflicts
to_remove = actions['remove']
for (request, pe) in to_remove:
logger.info('Removing conflicting package %s' % request)
try:
res = self.remove(request, force=True, only_priorities=only_priorities, only_if_not_process_running=only_if_not_process_running)
actions['errors'].extend(res['errors'])
actions['not_allowed'].extend(res.get('not_allowed', []))
if res['errors']:
print('Error removing %s:%s' % (request, ensure_unicode(res['errors'])))
if res['not_allowed']:
raise Exception('Removal of %s is not allowed' % repr(res['not_allowed']))
except Exception as e:
print('Error removing %s:%s' % (request, ensure_unicode(e)))
if not force:
raise
to_install = []
def is_process_running(processes):
processes = ensure_list(processes)
for p in processes:
if isrunning(p):
return p
return None
def is_allowed(package):
prio_allowed = only_priorities is None or package.priority in only_priorities
if not prio_allowed:
print('ERROR: Install of %s is not allowed at this stage because priority %s is not selected.' % (package.package, package.priority))
return False
install_process_blocked_by = only_if_not_process_running and package.impacted_process and is_process_running(package.impacted_process)
if install_process_blocked_by:
print('ERROR: Install of %s is not allowed at this stage because %s is running.' % (package.package, install_process_blocked_by))
return False
return True
for p in additional_install:
if is_allowed(p[1]):
to_install.append(p)
else:
not_allowed.append(p)
for p in to_upgrade:
if is_allowed(p[1]):
to_install.append(p)
else:
not_allowed.append(p)
for p in packages:
if is_allowed(p[1]):
to_install.append(p)
else:
not_allowed.append(p)
# get package entries to install to_install is a list of (request,package)
packages = [p[1] for p in to_install]
downloaded = self.download_packages(packages, usecache=usecache, printhook=printhook)
if downloaded.get('errors', []):
logger.critical('Error downloading some files : %s' % (downloaded['errors'],))
for request in downloaded.get('errors', []):
actions['errors'].append([request, None])
# check downloaded packages signatures and merge control data in local database
for fname in downloaded['downloaded'] + downloaded['skipped']:
pe = PackageEntry(waptfile=fname)
pe.check_control_signature(self.cabundle)
actions['downloads'] = downloaded
logger.debug('Downloaded : %s' % (downloaded,))
if not download_only:
# switch to manual mode
for (request, p) in skipped:
if request in apackages and not p.explicit_by:
logger.info('switch to manual mode for %s' % (request,))
self.waptdb.switch_to_explicit_mode(p.package, installed_by or self.user)
for (request, p) in to_install:
try:
if not p.localpath or not os.path.isfile(p.localpath):
raise EWaptDownloadError('Package file %s not downloaded properly.' % p.filename)
result = self.install_wapt(p.localpath,
params_dict=params_dict,
explicit_by=installed_by or self.user,
force=force
)
if result['install_status'] == 'OK':
if p.localpath.startswith(self.packages_cache_dir):
logger.info('Delete %s' % p.localpath)
os.remove(p.localpath)
if result:
for k in result.as_dict():
p[k] = result[k]
if not result or result['install_status'] != 'OK':
actions['errors'].append([request, p])
logger.critical('Package %s not installed due to errors' % (request,))
except Exception as e:
actions['errors'].append([request, p, ensure_unicode(traceback.format_exc())])
logger.critical('Package %s not installed due to errors : %s' % (request, ensure_unicode(e)))
if logger.level == logging.DEBUG:
raise
return actions
else:
logger.info('Download only, no install performed')
return actions
[docs] def download_packages(self, package_requests=List[str], usecache=True, printhook=None):
r"""Download a list of packages (requests are of the form packagename(=version) )
If several packages are matching a request, the highest/latest only is kept.
Args:
package_requests (str or list): list of packages to prefetch
usecache (boolean) : if True, don't download package if already in cache
printhook (func) : callback with signature report(received,total,speed,url) to display progress
Returns:
dict: with keys {"downloaded,"skipped","errors","packages"} and list of PackageEntry.
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> def nullhook(*args):
... pass
>>> wapt.download_packages(['tis-firefox','tis-waptdev'],usecache=False,printhook=nullhook)
{'downloaded': [u'c:/wapt\\cache\\tis-firefox_37.0.2-9_all.wapt', u'c:/wapt\\cache\\tis-waptdev.wapt'], 'skipped': [], 'errors': []}
"""
package_requests = self._ensure_package_requests_list(package_requests, keep_package_entries=True)
downloaded = []
skipped = []
errors = []
packages = []
for p in package_requests:
if isinstance(p, PackageRequest):
mp = self.waptdb.packages_matching(p)
if mp:
packages.append(mp[-1])
else:
errors.append((p, 'Unavailable package %s' % (p,)))
logger.critical('Unavailable package %s' % (p,))
elif isinstance(p, PackageEntry):
packages.append(p)
else:
raise Exception('Invalid package request %s' % p)
def report(received, total, speed, url):
self.check_cancelled()
try:
if total > 1:
stat = '%s : %i / %i (%.0f%%) (%s/s)\r' % (url, received, total, 100.0*received/total, format_bytes(speed))
print(stat)
else:
stat = ''
self.runstatus = 'Downloading %s : %s' % (entry.package, stat)
except:
self.runstatus = 'Downloading %s' % (entry.package,)
if not printhook:
printhook = report
for entry in packages:
self.check_cancelled()
if platform.system() == 'Windows':
target_dir = self.packages_cache_dir
else:
if os.geteuid() == 0:
target_dir = self.packages_cache_dir
else:
target_dir = os.path.join(os.path.expanduser("~"), "waptdev")
if not os.path.isdir(target_dir):
os.mkdir(target_dir)
# use specific repository settings for download
res = self.get_repo(entry.repo).download_packages(entry,
target_dir=target_dir,
usecache=usecache,
printhook=printhook)
downloaded.extend(res['downloaded'])
skipped.extend(res['skipped'])
errors.extend(res['errors'])
return {"downloaded": downloaded, "skipped": skipped, "errors": errors, "packages": packages}
[docs] def download_icons(self, package_requests, usecache=True, printhook=None):
r"""Download a list of package icons (requests are of the form packagename (>version) )
returns a dict of {"downloaded,"skipped","errors"}
Args:
package_requests (str or list): list of packages to prefetch
usecache (boolean) : if True, don't download package if already in cache
printhook (func) : callback with signature report(received,total,speed,url) to display progress
Returns:
dict: with keys {"downloaded,"skipped","errors","packages"} and list of PackageEntry.
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> def nullhook(*args):
... pass
>>> wapt.download_packages(['tis-firefox','tis-waptdev'],usecache=False,printhook=nullhook)
{'downloaded': [u'c:/wapt\\cache\\tis-firefox_37.0.2-9_all.wapt', u'c:/wapt\\cache\\tis-waptdev.wapt'], 'skipped': [], 'errors': []}
"""
package_requests = self._ensure_package_requests_list(package_requests, keep_package_entries=True)
downloaded = []
skipped = []
errors = []
packages = []
for p in package_requests:
if isinstance(p, PackageRequest):
mp = self.waptdb.packages_matching(p)
if mp:
packages.append(mp[-1])
else:
errors.append((p, 'Unavailable package %s' % (p,)))
logger.critical('Unavailable package %s' % (p,))
elif isinstance(p, PackageEntry):
packages.append(p)
else:
raise Exception('Invalid package request %s' % p)
for entry in packages:
self.check_cancelled()
def report(received, total, speed, url):
self.check_cancelled()
try:
if total > 1:
stat = '%s : %i / %i (%.0f%%) (%s/s)\r' % (url, received, total, 100.0*received/total, format_bytes(speed))
print(stat)
else:
stat = ''
self.runstatus = 'Downloading %s : %s' % (entry.package, stat)
except:
self.runstatus = 'Downloading %s' % (entry.package,)
"""
if not printhook:
printhook = report
"""
target_dir = os.path.join(self.packages_cache_dir, 'icons')
res = self.get_repo(entry.repo).download_icons(entry,
target_dir=target_dir,
usecache=usecache,
printhook=printhook)
downloaded.extend(res['downloaded'])
skipped.extend(res['skipped'])
errors.extend(res['errors'])
return {"downloaded": downloaded, "skipped": skipped, "errors": errors, "packages": packages}
[docs] def get_repo(self, repo_name):
for r in self.repositories:
if r.name == repo_name:
return r
return None
def _get_uninstallkeylist(self, uninstall_key_str):
"""Decode uninstallkey list from db field
For historical reasons, this field is encoded as str(pythonlist)
or sometimes simple repr of a str
..Changed 1.6.2.8:: uninstallkeylist is a json representation of list.
Returns:
list
"""
if uninstall_key_str:
if uninstall_key_str.startswith("['") or uninstall_key_str.startswith("[u'"):
# python encoded repr of a list
try:
# transform to a json like array.
guids = json.loads(uninstall_key_str.replace("[u'", "['").replace(", u'", ',"').replace("'", '"'))
except:
guids = uninstall_key_str
elif uninstall_key_str[0] in ["'", '"']:
# simple python string, removes quotes
guids = uninstall_key_str[1:-1]
else:
try:
# normal json encoded list
guids = ujson.loads(uninstall_key_str)
except:
guids = uninstall_key_str
if isinstance(guids, str):
guids = [guids]
return guids
else:
return []
[docs] def remove(self,
packages_list,
force=False,
only_priorities=None,
only_if_not_process_running=False):
"""Removes a package giving its package name, unregister from local status DB
Args:
packages_list (str or list or path): packages to remove (package name,
list of package requirement, package entry or development directory)
force : if True, unregister package from local status database, even if uninstall has failed
Returns:
dict: {'errors': [], 'removed': [], 'not_allowed': []}
"""
result = {'removed': [], 'errors': [], 'not_allowed': []}
if not isinstance(packages_list, list):
packages_list = [packages_list]
logger.info('Trying to remove %s with force=%s, only_priorities=%s, only_if_not_process_running=%s' % (repr(packages_list), force, only_priorities, only_if_not_process_running))
def is_process_running(processes):
processes = ensure_list(processes)
for p in processes:
if isrunning(p):
return p
return None
def is_allowed(dict_package):
if only_priorities is not None and dict_package.get('priority') in only_priorities:
print('Uninstall of %s not allowed because priority %s is not selected.' % (dict_package.get('package'), dict_package('priority')))
return False
uninstall_process_blocked_by = only_if_not_process_running and dict_package.get('impacted_process') and is_process_running(dict_package.get('impacted_process'))
if uninstall_process_blocked_by:
print('ERROR: Uninstall of %s is not allowed at this stage because %s is running.' % (dict_package.get('package'), uninstall_process_blocked_by))
return False
return True
for package in packages_list:
try:
self.check_cancelled()
# development mode, remove a package by its directory
if isinstance(package, str) and os.path.isfile(os.path.join(package, 'WAPT', 'control')):
package = PackageEntry().load_control_from_wapt(package).package
elif isinstance(package, PackageEntry):
package = package.package
else:
pe = self.is_installed(package)
if pe:
package = pe.package
q = self.waptdb.query("""\
select * from wapt_localstatus
where package=?
""", (package,))
if not q:
logger.debug("Package %s not installed, removal aborted" % package)
return result
# several versions installed of the same package... ?
for mydict in q:
# check that removal is allowed...
if not is_allowed(mydict):
result['not_allowed'].append(mydict['package'])
continue
self.runstatus = "Removing package %s version %s from computer..." % (mydict['package'], mydict['version'])
# removes recursively meta packages which are not satisfied anymore
additional_removes = self.check_remove(package)
cant_remove = False
for parent_package in additional_removes:
if not is_allowed(parent_package):
cant_remove = True
result['not_allowed'].append(mydict['package'])
break
if cant_remove:
logger.info('Removal of %s is not allowed at this stage because one parent package can not be removed' % mydict['package'])
result['not_allowed'].append(mydict['package'])
continue
if mydict.get('impacted_process', None):
killalltasks(ensure_list(mydict['impacted_process']))
if mydict['uninstall_key']:
# cook the uninstall_key because could be either repr of python list or string
# should be now json list in DB
uninstall_keys = self._get_uninstallkeylist(mydict['uninstall_key'])
if uninstall_keys:
for uninstall_key in uninstall_keys:
if setuphelpers.uninstall_key_exists(uninstall_key):
if sys.platform == 'win32':
try:
uninstall_cmd = self.uninstall_cmd(uninstall_key)
if uninstall_cmd:
logger.info('Launch uninstall cmd %s' % (uninstall_cmd,))
# if running processes, kill them before launching uninstaller
print(self.run(uninstall_cmd))
setuphelpers.wait_uninstallkey_absent(uninstall_key,max_loop=self.config.getint('global', 'uninstallkey_timeout'))
if setuphelpers.uninstall_key_exists(uninstall_key):
setuphelpers.error('Uninstallkey still present')
except Exception as e:
logger.critical("Critical error during uninstall: %s" % (ensure_unicode(e)))
result['errors'].append((package,traceback.format_exc()))
if not force:
raise
if sys.platform == 'darwin':
try:
if setuphelpers.uninstall_key_exists(uninstall_key):
if uninstall_key.startswith('pkgid:'):
setuphelpers.uninstall_pkg(uninstall_key[6:])
else:
if uninstall_key.startswith('/Applications/'):
setuphelpers.uninstall_app(uninstall_key[14:])
if setuphelpers.uninstall_key_exists(uninstall_key):
setuphelpers.error('Uninstallkey still present')
except Exception as e:
logger.critical("Critical error during uninstall: %s" % (ensure_unicode(e)))
result['errors'].append((package,traceback.format_exc()))
if not force:
raise
else:
logger.debug('uninstall key not registered in local DB status.')
if mydict['install_status'] != 'ERROR':
try:
self.uninstall(package)
except Exception as e:
logger.critical('Error running uninstall script: %s' % e)
result['errors'].append((package,traceback.format_exc()))
if not force:
raise
self.remove_package_persistent_dirs(mydict.get('package_uuid'))
logger.info('Remove status record from local DB for %s' % package)
if mydict['package_uuid']:
with self.waptdb, self.waptpublicdb:
self.waptdb.remove_install_status(package_uuid=mydict['package_uuid'])
self.waptpublicdb.execute('update wapt_publicstatus set uninstall_date=? where package_uuid=? and uninstall_date is NULL',[datetime2isodate(), mydict['package_uuid']])
else:
# backard
self.waptdb.remove_install_status(package=package)
result['removed'].append(package)
if reversed(additional_removes):
logger.info('Additional packages to remove : %s' % additional_removes)
for apackage in additional_removes:
res = self.remove(apackage, force=True)
result['removed'].extend(res['removed'])
result['errors'].extend(res['errors'])
return result
finally:
self.store_upgrade_status()
self.runstatus = ''
[docs] def host_packagename(self):
"""Return package name for current computer"""
# return "%s" % (setuphelpers.get_hostname().lower())
return "%s" % (self.host_uuid,)
[docs] def get_host_packages_names(self):
"""Return list of implicit host package names based on computer UUID and AD Org Units
Returns:
list: list of str package names.
"""
"""Return list of implicit available host packages based on computer UUID and AD Org Units
Returns:
list: list of PackageEntry.
"""
result = []
host_package = self.host_packagename()
result.append(host_package)
# ini configured profiles
if self.host_profiles:
result.extend([make_valid_package_name(p) for p in self.host_profiles])
previous_dn_part_type = ''
host_dn = self.host_dn
if host_dn and pyldap:
dn_parts = [elem[0]+'='+elem[1] for elem in pyldap.parse_dn(host_dn)]
for i in range(1, len(dn_parts)):
dn_part = dn_parts[i]
dn_part_type, value = dn_part.split('=', 1)
if dn_part_type.lower() == 'dc' and dn_part_type == previous_dn_part_type:
break
level_dn = ','.join(dn_parts[i:])
# spaces and
result.append(make_valid_package_name(level_dn))
previous_dn_part_type = dn_part_type
return result
[docs] def get_host_packages(self):
"""Return list of implicit available host packages based on computer UUID and AD Org Units
Returns:
list: list of PackageEntry.
"""
result = []
package_names = self.get_host_packages_names()
for pn in package_names:
packages = self.is_available(pn)
if packages and packages[-1].section in ('host', 'unit', 'profile'):
result.append(packages[-1])
return result
[docs] def get_outdated_host_packages(self):
"""Check and return the available host packages available and not installed"""
result = []
host_packages = self.get_host_packages()
logger.debug('Checking availability of host packages "%s"' % (host_packages, ))
for package in host_packages:
if self.is_locally_allowed_package(package):
logger.debug('Checking if %s is installed/outdated' % package.asrequirement())
installed_package = self.is_installed(package.package)
if not installed_package or installed_package < package:
result.append(package)
return result
[docs] def get_installed_host_packages(self):
"""Get the implicit package names (host and unit packages) which are installed but no longer relevant
Returns:
list: of installed package names
"""
return [p.package for p in self.installed(True) if p.section in ('host', 'unit', 'profile')]
[docs] def get_unrelevant_host_packages(self):
"""Get the implicit package names (host and unit packages) which are installed but no longer relevant
Returns:
list: of installed package names
"""
installed_host_packages = self.get_installed_host_packages()
expected_host_packages = self.get_host_packages_names()
return [pn for pn in installed_host_packages if pn not in expected_host_packages]
[docs] def upgrade(self, only_priorities=None, only_if_not_process_running=False):
"""Install "well known" host package from main repository if not already installed
then query localstatus database for packages with a version older than repository
and install all newest packages
Args:
priorities (list of str): If not None, upgrade only packages with these priorities.
Returns:
dict: {'upgrade': [], 'additional': [], 'downloads':
{'downloaded': [], 'skipped': [], 'errors': []},
'remove': [], 'skipped': [], 'install': [], 'errors': [], 'unavailable': []}
"""
try:
self.runstatus = 'Upgrade system'
upgrades = self.list_upgrade()
logger.debug('upgrades : %s' % upgrades)
result = dict(
install=[],
upgrade=[],
additional=[],
remove=[],
errors=[])
if upgrades['remove']:
self.runstatus = 'Removes outdated / conflicted packages'
result = merge_dict(result, self.remove(upgrades['remove'], force=True))
for key in ['additional', 'upgrade', 'install']:
self.runstatus = 'Install %s packages' % key
if upgrades[key]:
result = merge_dict(result, self.install(upgrades[key], process_dependencies=True))
result = merge_dict(result, self.install(list(upgrades.keys()), force=True, only_priorities=only_priorities, only_if_not_process_running=only_if_not_process_running))
self.store_upgrade_status()
# merge results
return result
finally:
self.runstatus = ''
[docs] def list_upgrade(self,current_datetime = None):
"""Returns a list of package requirements for packages which should be installed / upgraded / removed
Returns:
dict: {'additional': [], 'install': [], 'remove': [], 'upgrade': []}
"""
result = dict(
install=[],
upgrade=[],
additional=[],
remove=[],
immediate_installs=[])
# only most up to date (first one in list)
# put 'host' package at the end.
for current,next in self.waptdb.upgradeable_status():
if not current.section in ('host', 'unit', 'profile'):
result['upgrade'].append(next.asrequirement())
#result['install'].append()
to_remove = self.get_unrelevant_host_packages()
result['remove'].extend(to_remove)
if self.use_hostpackages:
host_packages = self.get_outdated_host_packages()
if host_packages:
for p in host_packages:
if self.is_locally_allowed_package(p):
req = p.asrequirement()
if not req in result['install']+result['upgrade']+result['additional']:
result['install'].append(req)
# get additional packages to install/upgrade based on new upgrades
depends = self.check_depends(result['install']+result['upgrade']+result['additional'])
if not current_datetime:
current_datetime = datetime2isodate()
# to not force install packages which can't be installed properly.
install_errors_packages_uuid = [p['package_uuid'] for p in self.waptdb.query("select package_uuid from wapt_localstatus where install_status='ERROR'")]
for l in ('upgrade', 'additional', 'install'):
for (r, candidate) in depends[l]:
req = candidate.asrequirement()
if not req in result['install']+result['upgrade']+result['additional']:
result[l].append(req)
if candidate and candidate.forced_install_on and not candidate.package_uuid in install_errors_packages_uuid and candidate.forced_install_on <= current_datetime and not candidate.package_uuid in result['immediate_installs']:
# explicit package_uuid request
result['immediate_installs'].append('{%s}' % candidate.package_uuid)
result['remove'].extend([p[1].asrequirement() for p in depends['remove'] if p[1].package not in result['remove']])
return result
[docs] def search(self, searchwords=[], exclude_host_repo=True, section_filter=None, newest_only=False, with_install_status=False):
"""Returns a list of packages which have the searchwords in their description
Args:
searchwords (str or list): words to search in packages name or description
exclude_host_repo (boolean): if True, don't search in host repoisitories.
section_filter (str or list): restrict search to the specified package sections/categories
Returns:
list: list of PackageEntry
"""
available = self.waptdb.packages_search(searchwords=searchwords, exclude_host_repo=exclude_host_repo, section_filter=section_filter)
if with_install_status:
installed = {p.package_uuid: p for p in self.waptdb.installed(include_errors=True, include_setup=False, include_control=False)}
upgradable = self.waptdb.upgradeable()
for p in available:
if p.package_uuid in installed:
current = installed[p.package_uuid]
if p == current:
p['installed'] = current
if p.package in upgradable:
p['status'] = 'U'
else:
p['status'] = 'I'
else:
p['installed'] = None
p['status'] = '-'
else:
p['installed'] = None
p['status'] = '-'
if newest_only:
filtered = []
last_package_ident = None
for package in sorted(available, reverse=True, key=self.packages_filter_for_host().get_package_compare_key):
if package.package_ident() != last_package_ident:
filtered.append(package)
last_package_ident = package.package_ident()
return list(reversed(filtered))
else:
return available
[docs] def list(self, searchwords=[]):
"""Returns a list of installed packages which have the searchwords
in their description
Args:
searchwords (list): list of words to llokup in package name and description
only entries which have words in the proper order are returned.
Returns:
list: list of PackageEntry matching the search words
>>> w = Wapt()
>>> w.list('zip')
[PackageEntry('tis-7zip','16.4-8') ]
"""
return self.waptdb.installed_search(searchwords=searchwords,)
[docs] def check_downloads(self, apackages=None, usecache=True):
"""Return list of available package entries
to match supplied packages requirements
Args:
apackages (list or str): list of packages
usecache (bool) : returns only PackageEntry not yet in cache
Returns:
list: list of PackageEntry to download
"""
result = []
if apackages is None:
actions = self.list_upgrade()
apackages = actions['install']+actions['additional']+actions['upgrade']
elif isinstance(apackages, str):
apackages = ensure_list(apackages)
elif isinstance(apackages, list):
# ensure that apackages is a list of package requirements (strings)
new_apackages = []
for p in apackages:
if isinstance(p, PackageEntry):
new_apackages.append(p.asrequirement())
else:
new_apackages.append(p)
apackages = new_apackages
for p in apackages:
entries = self.is_available(p)
if entries:
# download most recent
entry = entries[-1]
fullpackagepath = os.path.join(self.packages_cache_dir, entry.make_package_filename())
if usecache and (os.path.isfile(fullpackagepath) and os.path.getsize(fullpackagepath) == entry.size):
# check version
try:
cached = PackageEntry()
cached.load_control_from_wapt(fullpackagepath, calc_md5=False)
if entry != cached:
result.append(entry)
except Exception as e:
logger.warning('Unable to get version of cached package %s: %s' % (fullpackagepath, ensure_unicode(e),))
result.append(entry)
else:
result.append(entry)
else:
logger.debug('check_downloads : Package %s is not available' % p)
return result
[docs] def download_upgrades(self):
"""Download packages that can be upgraded"""
self.runstatus = 'Download upgrades'
try:
to_download = self.check_downloads()
return self.download_packages(to_download)
finally:
self.runstatus = ''
[docs] def authorized_certificates(self):
"""return a list of autorized package certificate issuers for this host
check_certificates_validity enable date checking.
"""
return [c for c in self.cabundle.trusted_certificates() if not self.check_certificates_validity or c.is_valid()]
[docs] def trust_package_signer(self,cert):
"""Add a certificate to the list of trusted package signers certificates
Stores the PEM encoded data in public_certs_dir directory.
The certificate is added to this current Wapt.cabundle instance for immediate use (until config is reloaded)
"""
added = self.cabundle.add_certificates(cert,trusted=True)
for signer_cert in added:
if signer_cert.public_cert_filename and os.path.isfile(signer_cert.public_cert_filename):
shutil.copyfile(signer_cert.public_cert_filename,os.path.join(self.public_certs_dir,os.path.basename(signer_cert.public_cert_filename)))
else:
signer_cert.save_as_pem(os.path.join(self.public_certs_dir,signer_cert.fingerprint))
[docs] def untrust_package_signer(self,cert):
"""Removes a certificate from the list of trusted package signers certificates based on the fingerprint of the certificate.
Certs in ssl public_certs_dir with matching fingerprint are deleted.
"""
removed = self.cabundle.remove_certificates(cert)
# removed all crt files in ssl which have the fingerprint of actually untrusted certs
for fn in glob.glob(os.path.join(self.public_certs_dir, '*.crt'))+glob.glob(os.path.join(self.public_certs_dir, '*.pem')):
try:
old_cert = SSLCertificate(crt_filename=fn)
for crt in removed:
if old_cert.fingerprint == crt.fingerprint:
os.unlink(fn)
break
except ValueError:
pass
[docs] def register_computer(self, description=None, retry_with_password=False):
"""Send computer informations to WAPT Server
if description is provided, updates local registry with new description
Returns:
dict: response from server.
>>> wapt = Wapt()
>>> s = wapt.register_computer()
>>>
"""
if not self.waptserver:
raise EWaptException('Unable to register: waptserver not defined')
if not self.waptserver.available():
raise EWaptException('Unable to register: waptserver %s not available' % self.waptserver.server_url)
if description:
self.write_param('host_description',description, public=True)
try:
setuphelpers.set_computer_description(description)
except Exception as e:
logger.info('Unable to change system computer description to %s: %s' % (description, e))
if self.waptserver and self.waptserver.available():
# force recalc uuid
self._host_uuid = None
# full inventory
new_hashes = {}
old_hashes = {}
inv = self._get_host_status_data(old_hashes, new_hashes, force=True, included_keys=['wapt_status','host_capabilities'])
inv['status_hashes'] = new_hashes
inv['uuid'] = self.host_uuid
inv['host_certificate'] = self.create_or_update_host_certificate()
inv['host_certificate_signing_request'] = self.get_host_certificate_signing_request().as_pem()
data = jsondump(inv).encode('utf8')
if not self.waptserver.use_kerberos:
urladdhost = 'add_host'
else:
urladdhost = 'add_host_kerberos'
signature = self.sign_host_content(data)
try:
result = self.waptserver.post(urladdhost,
data=data,
signature=signature,
signer=self.get_host_certificate().cn
)
except requests.HTTPError as e:
if e.response.status_code in (403, 404) and urladdhost == 'add_host_kerberos' and retry_with_password: # retry without kerberos
# retry without kerberos auth
result = self.waptserver.post('add_host',
data=data,
signature=signature,
signer=self.get_host_certificate().cn
)
elif e.response.status_code in (400, 401): # could be a bad certificate error, so retry without client side cert
# retry without ssl client auth
result = self.waptserver.post(urladdhost,
data=data,
signature=signature,
signer=self.get_host_certificate().cn,
use_ssl_auth=False
)
else:
result = dict(
success=False,
msg='Error sending registration data: status %s' % (e.response.status_code,)
)
if result and result['success']:
# if server has changed, reset cached inventory hash_status
if result['result'].get('server_uuid') != self.read_param('server_uuid', public=True):
self.delete_param('last_update_server_hashes')
self.delete_param('last_audit_data_server_date')
self.write_param('server_uuid', result['result'].get('server_uuid'), public=True)
self.write_param('last_successful_register', datetime2isodate(), public=True)
result_data = result.get('result', {})
if 'status_hashes' in result_data:
# invalidate unmatching hashes for next round.
self.write_param('last_update_server_hashes', result_data['status_hashes'])
# stores last_audit_data_server_date to send only newer data on next update server status.
self.write_param('last_audit_data_server_date',result_data.get('last_audit_data_server_date'))
if 'host_certificate' in result_data:
# server has signed the certificate, we replace our self signed one.
new_host_cert = SSLCertificate(crt_string=result_data['host_certificate'].encode('utf8'))
tasks_logger.info('Got signed certificate from server. Issuer: %s. CN: %s' % (new_host_cert.issuer_cn, new_host_cert.cn))
host_key = self.get_host_key()
if new_host_cert.cn == self.host_uuid and new_host_cert.match_key(host_key):
# be sure we have on disk the current host key.
tasks_logger.info('Save host key to %s' % (self.get_host_key_filename(),))
host_key.save_as_pem(filename=self.get_host_key_filename())
tasks_logger.info('Save host cert to %s' % (self.get_host_certificate_filename(),))
open(self.get_host_certificate_filename(),'w').write(result_data['host_certificate'])
"""
p12 = SSLPKCS12(filename = os.path.join(self.private_dir, self.host_uuid+'.p12'))
p12.certificate = new_host_cert
p12.private_key = host_key
p12.save_as_p12(friendly_name=self.host_uuid)
"""
self._host_certificate = None
self._host_certificate_timestamp = None
self.delete_param('host_certificate_fingerprint')
self.delete_param('host_certificate_authority_key_identifier')
# use newly signed client side auth certificate
self.set_client_cert_auth(self.waptserver,force=True)
for repo in self.repositories:
self.set_client_cert_auth(repo,force=True)
else:
tasks_logger.info('Signed certificate %s does not match host uuid %s or key.' % (new_host_cert.cn,self.host_uuid))
else: # register is not successful
self.delete_param('last_successful_register')
return result
else:
if not self.waptserver:
return dict(
success=False,
msg='Error sending registration: No WAPT server defined'
)
else:
return dict(
success=False,
msg='Error sending registration: WAPT server %s not available' % self.waptserver.server_url
)
[docs] def unregister_computer(self):
"""Remove computer informations from WAPT Server
Returns:
dict: response from server.
>>> wapt = Wapt()
>>> s = wapt.unregister_computer()
>>>
"""
if self.waptserver:
data = jsondump({'uuids': [self.host_uuid], 'delete_packages': 1, 'delete_inventory': 1}).encode('utf8')
result = self.waptserver.post('api/v3/hosts_delete',
data=data,
signature=self.sign_host_content(data),
signer=self.get_host_certificate().cn
)
if result and result['success']:
self.delete_param('last_update_server_hashes')
self.delete_param('last_audit_data_server_date')
self.delete_param('server_uuid')
if os.path.isfile(self.get_host_certificate_filename()):
os.unlink(self.get_host_certificate_filename())
return result
else:
return dict(
success=False,
msg='No WAPT server defined',
data={},
)
[docs] def get_host_key_filename(self):
"""Full filepath of the host own RSA private key
Returns:
str
"""
return os.path.join(self.private_dir, self.host_uuid+'.pem')
[docs] def get_host_certificate_filename(self):
"""Full filepath of the host own certificate and RSA public key
Returns:
str
"""
return os.path.join(self.private_dir, self.host_uuid+'.crt')
[docs] def get_host_certificate(self):
"""Return the current host certificate.
If the certificate does not yet exist, it is created as a self signed certificate
and stored in get_host_certificate_filename path
Returns:
SSLCertificate: host public certificate.
"""
cert_fn = self.get_host_certificate_filename()
if not self._host_certificate or not os.path.isfile(cert_fn) or self._host_certificate_timestamp != os.stat(cert_fn).st_mtime:
if not os.path.isfile(cert_fn):
self.create_or_update_host_certificate()
self._host_certificate = SSLCertificate(cert_fn)
self._host_certificate_timestamp = os.stat(cert_fn).st_mtime
return self._host_certificate
[docs] def get_host_certificate_signing_request(self):
"""Return a CSR with CN and AltSubjectNames pointing to this host uuid
Returns:
SSLCertificateSigningRequest: host public certificate sigbinbg request.
"""
host_key = self.get_host_key()
csr = host_key.build_csr(
cn=self.host_uuid,
altnames=[setuphelpers.get_hostname()],
is_ca=False,
is_code_signing=False,
is_client_auth=True,
key_usages=['digital_signature', 'content_commitment', 'data_encipherment', 'key_encipherment'])
return csr
[docs] def create_or_update_host_certificate(self, force_recreate=False):
"""Create a rsa key pair for the host and a x509 certiticate.
Location of key is <wapt_root>\private
Should be kept secret
restricted access to system account and administrators only.
Args:
force_recreate (bool): recreate key pair even if already exists for this FQDN.
Returns:
str: x509 certificate of this host.
"""
crt_filename = self.get_host_certificate_filename()
if force_recreate or not os.path.isfile(crt_filename):
logger.info('Creates host keys pair and x509 certificate %s' % crt_filename)
self._host_key = self.get_host_key()
if not os.path.isdir(self.private_dir):
os.makedirs(self.private_dir)
crt = self._host_key.build_sign_certificate(
ca_signing_key=None,
ca_signing_cert=None,
cn=self.host_uuid,
altnames=[setuphelpers.get_hostname()],
is_ca=True,
is_code_signing=False,
is_client_auth=True)
crt.save_as_pem(crt_filename)
self.delete_param('host_certificate_fingerprint')
self.delete_param('host_certificate_authority_key_identifier')
# check validity
with open(crt_filename, 'rb') as f:
return f.read()
[docs] def get_host_key(self, create=True):
"""Return private key used to sign uploaded data from host
Create key if it does not exists yet.
Returns:
SSLPrivateKey: Private key used to sign data posted by host.
"""
key_filename = self.get_host_key_filename()
if self._host_key is None or not os.path.isfile(key_filename) or self._host_key_timestamp != os.stat(key_filename).st_mtime:
# create keys pair / certificate if not yet initialised
if create and not os.path.isfile(key_filename):
self._host_key = SSLPrivateKey(key_filename)
self._host_key.create()
if not os.path.isdir(os.path.dirname(key_filename)):
os.makedirs(os.path.dirname(key_filename))
self._host_key.save_as_pem()
elif os.path.isfile(key_filename):
self._host_key = SSLPrivateKey(key_filename)
self._host_key_timestamp != os.stat(key_filename).st_mtime
return self._host_key
[docs] def sign_host_content(self, data):
"""Sign data str with host private key with sha256 + RSA
Args:
data (bytes) : data to sign
Returns
bytes: signature of sha256 hash of data.
"""
key = self.get_host_key()
return key.sign_content(hexdigest_for_data(data, md='sha256'))
[docs] def get_last_update_status(self):
"""Get update status of host as stored at the end of last operation.
Returns:
dict:
'date': timestamp of last operation
'runstatus': last printed message of wapt core
'running_tasks': list of tasks
'errors': list of packages not installed properly
'upgrades': list of packages which need to be upgraded
"""
status = self.read_param('last_update_status', {"date": "", "running_tasks": [], "errors": [], "upgrades": [],"immediate_installs": []}, ptype='json')
status['runstatus'] = self.read_param('runstatus', '')
status['audit_status'] = self.waptdb.audit_status()
return status
def _get_package_status_rowid(self, package_entry=None, package_name=None):
"""Return ID of package_status record for package_name
Args:
package_entry (PackageEntry): package entry to lookup by package_uuid or name
package_name (str): explicit package name
# todo: should be a PackageRequest
Returns:
int: rowid in local wapt_localstatus table
"""
with self.waptdb as waptdb:
if package_entry is not None and package_entry.package_uuid:
cur = waptdb.execute("""select rowid from wapt_localstatus where package_uuid=?""", (package_entry.package_uuid,))
else:
cur = waptdb.execute("""select rowid from wapt_localstatus where package=?""", (package_entry.package if package_entry is not None else package_name,))
pe = cur.fetchone()
if not pe:
return None
else:
return pe[0]
[docs] def update_package_install_status(self, **kwargs):
"""Update the install status
"""
return self.waptdb.update_install_status(**kwargs)
[docs] def get_cache_domain_info(self, force=False):
last_result = self.read_param('domain_info')
if not last_result:
last_result = {'ou': '', 'site': '', 'groups': []}
last_date = self.waptdb.get_param('last_domain_info_date', ptype='datetime')
now = datetime.datetime.utcnow()
if last_date:
delta = now - last_date
else:
force = True
maxdelta = 60 * 60 * 2
if force or (delta.seconds > maxdelta):
try:
if sys.platform == 'win32':
if self.use_ad_groups:
last_result = {'groups': setuphelpers.get_computer_groups()}
else:
last_result = {'groups': []}
else:
last_result = setuphelpers.get_domain_info()
except:
last_result = last_result
self.save_last_domain_info_date(now)
self.save_domain_info(last_result)
return last_result
def _get_new_audit_data(self,force=False):
last_audit_data_server_date = self.read_param('last_audit_data_server_date')
if force or not last_audit_data_server_date:
last_audit_data_server_date = None
if last_audit_data_server_date and last_audit_data_server_date > datetime2isodate():
last_audit_data_server_date = datetime2isodate()
return self.read_audit_data_since(last_audit_data_server_date)
[docs] def get_cached_packages_uuids(self):
r = WaptLocalRepo(self.packages_cache_dir)
r.update_packages_index(include_certificates=False,include_crls=False,extract_icons=False)
return [p.package_uuid for p in r.packages()]
def _get_host_status_data(self, old_hashes, new_hashes, force=False, included_keys=None, excluded_keys=[]):
"""Build the data to send to server where update_server_status required
Returns:
dict
"""
def _default_data_state(data):
return hashlib.sha1(pickle.dumps(data)).hexdigest()
def _add_data_if_updated(inv, key, data, old_hashes, new_hashes, force, data_state_func=_default_data_state):
"""Add the data to inv as key if modified since last update_server_status
if data is None, nothing is added.
if data_state_func is None, data is always added.
"""
if data is not None:
if data_state_func is not None:
newhash = data_state_func(data)
oldhash = old_hashes.get(key, None)
if force or oldhash != newhash:
if isinstance(data,dict):
data['updated_on'] = datetime2isodate()
inv[key] = data
new_hashes[key] = newhash
else:
inv[key] = data
if isinstance(data,dict):
data['updated_on'] = datetime2isodate()
elif data_state_func is None:
inv[key] = data
if isinstance(data,dict):
data['updated_on'] = datetime2isodate()
def _get_host_info():
host_info = setuphelpers.host_info()
# optionally forced dn
host_info['computer_ad_dn'] = self.host_dn
host_info['computer_ad_site'] = self.host_site
if not host_info.get('description'):
host_info['description'] = self.read_param('host_description', public=True)
host_info['repositories'] = ";".join([r.as_dict()['repo_url'] for r in self.repositories if not(r.as_dict()['repo_url'].endswith('-host'))])
if self.use_ad_groups:
host_info['computer_ad_groups'] = self.get_cache_domain_info()['groups']
return host_info
def _get_authorized_certificates_pems():
return [c.as_pem() for c in self.authorized_certificates()]
def _config_overview():
return config_overview(self.wapt_base_dir, self.config_filename)
def _wmi_info():
return setuphelpers.wmi_info(exclude_subkeys=['OEMLogoBitmap','PrinterPaperNames','PaperSizesSupported'])
def _packages_audit_status_delta():
if not force:
last_packages_audit_status_server_date = self.read_param('last_packages_audit_status_server_date')
else:
last_packages_audit_status_server_date = None
result = self.waptdb.packages_audit_inventory(after_date=last_packages_audit_status_server_date)
if len(result)>1:
return result
else:
return None
def _packages_install_status_delta():
# for backward compat
if force or not old_hashes.get('installed_packages_ids'):
since_revision=None
else:
since_revision = old_hashes.get('installed_packages')
if since_revision and not isinstance(since_revision,int):
since_revision = None
result = self.waptdb.installed_packages_inventory(since_status_revision=since_revision)
if len(result)>1:
return result
else:
return None
def _audit_data():
result = list(self._get_new_audit_data(force=force))
if len(result) > 1:
return result
else:
return None
inv = {
'uuid': self.host_uuid,
'computer_fqdn': ensure_unicode(setuphelpers.get_hostname())
}
status_revision = self.waptdb.get_status_revision()
inv['status_revision'] = status_revision
timing_store = {}
data_defs = {
'host_info': _get_host_info,
'host_networking': setuphelpers.host_info_networking,
'host_capabilities': self.host_capabilities,
'host_metrics': setuphelpers.host_metrics,
'wapt_status': self.wapt_status,
'installed_softwares': self.merge_installed_softwares_and_wua_list,
'installed_packages_ids': self.waptdb.installed_packages_ids,
'installed_packages': (_packages_install_status_delta, lambda d: status_revision),
'packages_audit_status': _packages_audit_status_delta,
'last_update_status': self.get_last_update_status,
'authorized_certificates': _get_authorized_certificates_pems,
'configurations': _config_overview,
'audit_data': _audit_data,
#'host_info.list_services': setuphelpers.service_list,
#'host_info.listening_sockets': setuphelpers.listening_sockets,
}
if os.name == 'nt':
if self.is_enterprise():
try:
wua_client = self.waptwua()
data_defs['wuauserv_status'] = wua_client.get_wuauserv_status
data_defs['waptwua_status'] = wua_client.stored_waptwua_status
data_defs['waptwua_updates'] = wua_client.stored_updates
data_defs['waptwua_updates_localstatus'] = wua_client.stored_updates_localstatus
#data_defs['waptwua_rules_packages'] = wua_client.stored_waptwua_rules
except Exception as e:
logger.critical('Unable to get waptwua status: %s' % e)
if included_keys is None:
included_keys = [k for k in data_defs.keys() if not k in excluded_keys]
# populate inventory
for key in included_keys:
if not key in excluded_keys:
with Timeit(key,store=timing_store):
try:
if isinstance(data_defs[key],tuple):
# specific function to check data version
_add_data_if_updated(inv,key, data_defs[key][0](), old_hashes, new_hashes, force=force, data_state_func=data_defs[key][1])
else:
_add_data_if_updated(inv,key, data_defs[key](), old_hashes, new_hashes, force=force)
except Exception as e:
logger.critical('Unable to build status data for key %s: %s' % (key,repr(e)))
return inv
[docs] def build_update_status_data(self, force=False, excluded_keys = []):
# avoid sending data to the server if it has not been updated.
new_hashes = {}
old_hashes = self.read_param('last_update_server_hashes', {}, ptype='json')
inv = self._get_host_status_data(old_hashes, new_hashes, force=force, excluded_keys=excluded_keys)
inv['status_hashes'] = new_hashes
return inv
[docs] def update_server_status(self, force=False, excluded_keys=[], errors = None):
"""Send host_info, installed packages and installed softwares,
and last update status informations to WAPT Server,
but don't send register info like dmi or wmi.
.. versionchanged:: 1.4.3
if last status has been properly sent to server and data has not changed,
don't push data again to server.
the hash is stored in memory, so is not pass across threads or processes.
>>> wapt = Wapt()
>>> s = wapt.update_server_status()
>>>
"""
result = None
sys.stdout.flush()
try_register = False
retry = True
if not self.waptserver:
return None
if not self.waptserver_available():
tasks_logger.info('WAPT Server is not available to store current host status')
return None
while retry:
try:
inv = self.build_update_status_data(force=force, excluded_keys=excluded_keys)
new_hashes = inv['status_hashes']
logger.info('Updated data keys : %s' % [k for k in new_hashes if k != new_hashes.get(k)])
logger.info('Supplied data keys : %s' % list(inv.keys()))
data = jsondump(inv).encode('utf8')
logger.info('Sending %s bytes to server' % len(data))
signature = self.sign_host_content(data)
try:
result = self.waptserver.post('update_host',
data=data,
signature=signature,
signer=self.get_host_certificate().cn
)
if result and result['success']:
retry = False
# stores for next round.
self.write_param('server_uuid', result['result'].get('server_uuid'), public=True)
self.write_param('last_update_server_status_timestamp', datetime.datetime.utcnow())
# stores last_audit_data_server_date to send only newer data on next update server status.
last_audit_data_server_date = result['result'].get('last_audit_data_server_date')
if last_audit_data_server_date is not None:
self.write_param('last_audit_data_server_date',last_audit_data_server_date)
last_packages_audit_status_server_date = result['result'].get('last_packages_audit_status_server_date')
if last_packages_audit_status_server_date is not None:
self.write_param('last_packages_audit_status_server_date',last_packages_audit_status_server_date)
if 'status_hashes' in result.get('result', {}):
# known server hashes for next round.
self.write_param('last_update_server_hashes', result['result']['status_hashes'])
logger.info('Status on server %s updated properly' % self.waptserver.server_url)
else:
logger.info('Error updating Status on server %s: %s' % (self.waptserver.server_url, result and result['msg'] or 'No message'))
if result.get('error_code') in ('ewaptmissingcertificate','ewaptbadserverauthentication'):
try_register = True
retry = True
else:
retry = False
except requests.HTTPError as e:
error = 'Unable to update server status : %s' % (e,)
if errors:
errors.append(error)
logger.warning(error)
if e.response.status_code in (400, 401, 403):
last_successful_register = self.read_param('last_successful_register')
if not last_successful_register or (datetime.datetime.utcnow() - isodate2datetime(last_successful_register) > datetime.timedelta(hours=2)):
try_register = True
else:
retry=False
else:
retry = False
except EWaptBadServerAuthentication as e:
error = 'Unable to update server status : %s' % (e,)
if errors:
errors.append(error)
logger.warning(error)
try_register = True
if result and result['success']:
db_data = result.get('result')
tasks_logger.info('update_server_status successful (data size:%s, keys:%s)' % (len(data),inv.keys()))
else:
db_data = None
if try_register or (db_data and db_data.get('computer_fqdn', None).lower() != setuphelpers.get_hostname().lower()):
try_register = False
tasks_logger.warning('Host on the server is not known or not known under this FQDN name (known as %s). Trying to register the computer...' % (db_data and db_data.get('computer_fqdn', None) or None))
result = self.register_computer()
if result and result['success']:
tasks_logger.info('New registration successful. Retring sending host status.')
self.reload_config_if_updated()
retry = True
else:
logger.critical('Unable to register: %s' % result and result['msg'])
retry = False
except Exception as e:
error = 'Unable to update server status : %s' % (e,)
if errors:
errors.append(error)
tasks_logger.warning(error)
logger.debug(traceback.format_exc())
break
return result
[docs] def waptserver_available(self):
"""Test reachability of waptserver.
If waptserver is defined and available, return True, else False
Returns:
boolean: True if server is defined and actually reachable
"""
return self.waptserver and self.waptserver.available()
[docs] def inc_status_revision(self, inc=1):
rev = self.read_param('status_revision', 0, ptype='int')+inc
self.write_param('status_revision', rev)
return rev
[docs] def merge_installed_softwares_and_wua_list(self):
soft_inventory = setuphelpers.installed_softwares()
for s in soft_inventory:
# append win32 for uniqueness on windows
s['software_id'] = s['key'] + {False:'##win32',True:'',None:''}.get(s.get('win64'))
if self.waptwua_enabled and self.is_enterprise():
dict_kb_name = self.waptdb.get_param('waptwua.simple.list')
if dict_kb_name:
for u in dict_kb_name:
soft_inventory.append({'software_id':u,
'key': u,
'name': '%s (%s)' % (u, dict_kb_name[u]),
'version': u.replace('KB', ''),
'install_date': '',
'install_location': '',
'uninstall_string': '',
'publisher': 'Microsoft',
'system_component': 1,
'win64': setuphelpers.iswin64() })
return soft_inventory
[docs] def wapt_status(self):
"""Wapt configuration and version informations
Returns:
dict: versions of main main files, waptservice config,
repos and waptserver config
>>> w = Wapt()
>>> w.wapt_status()
{
'setuphelpers-version': '1.1.1',
'waptserver': {
'wapt_server': u'tranquilit.local',
'proxies': {
'http': None,
'https': None
},
'server_url': 'https: //wapt.tranquilit.local'
},
'waptservice_protocol': 'http',
'repositories': [{
'wapt_server': u'tranquilit.local',
'proxies': {
'http': None,
'https': None
},
'name': 'global',
'repo_url': 'http: //wapt.tranquilit.local/wapt'
},
{
'wapt_server': u'tranquilit.local',
'proxies': {
'http': None,
'https': None
},
'name': 'wapt-host',
'repo_url': 'http: //srvwapt.tranquilit.local/wapt-host'
}],
'common-version': '1.1.1',
'wapt-exe-version': u'1.1.1.0',
'waptservice_port': 8088,
'wapt-py-version': '1.1.1'
}
"""
result = {}
if os.name == 'nt':
waptexe = os.path.join(self.wapt_base_dir, 'wapt-get.exe')
if os.path.isfile(waptexe):
result['wapt-get-version'] = setuphelpers.get_file_properties(waptexe)['FileVersion']
with open(os.path.join(self.wapt_base_dir, 'version-full'), 'r') as wapt_version_full:
result['wapt-version-full'] = wapt_version_full.readline().rstrip()
result['waptutils-version'] = __version__
trusted_certs_sha256 = []
trusted_certs_cn = []
invalid_certs_sha256 = []
result['last_external_ip'] = self.waptdb.get_param('last_external_ip')
for c in self.authorized_certificates():
try:
for c2 in self.cabundle.check_certificates_chain(c):
if not c2.fingerprint in trusted_certs_sha256:
trusted_certs_sha256.append(c2.fingerprint)
trusted_certs_cn.append(c2.cn)
except Exception as e:
logger.warning('Certificate %s invalid (fingerprint %s expiration %s): %s' % (c.cn, c.fingerprint, c.not_after, e))
invalid_certs_sha256.append(c.fingerprint)
result['authorized_certificates_sha256'] = trusted_certs_sha256
result['invalid_certificates_sha256'] = invalid_certs_sha256
result['authorized_certificates_cn'] = trusted_certs_cn
result['maturities'] = self.maturities
result['locales'] = self.locales
result['is_remote_repo'] = self.config.getboolean('repo-sync', 'enable_remote_repo') if (self.config.has_section('repo-sync') and self.config.has_option('repo-sync', 'enable_remote_repo')) else False
result['remote_reboot_allowed'] = self.config.getboolean('global', 'allow_remote_reboot') if self.config.has_option('global', 'allow_remote_reboot') else False
result['remote_shutdown_allowed'] = self.config.getboolean('global', 'allow_remote_shutdown') if self.config.has_option('global', 'allow_remote_shutdown') else False
result['remote_repo_url'] = '' if not(result['is_remote_repo']) else self.config.get('repo-sync', 'remote_repo_url') if (self.config.has_section('repo-sync') and self.config.has_option('repo-sync', 'remote_repo_url')) else 'https://'+setuphelpers.get_fqdn()+'/wapt'
result['wol_relay'] = self.config.getboolean('global', 'wol_relay') if self.config.has_option('global', 'wol_relay') else result['is_remote_repo']
result['use_repo_rules'] = self.use_repo_rules
if sys.platform == 'win32':
result['pending_reboot_reasons'] = setuphelpers.pending_reboot_reasons()
# read from config
if self.config.has_option('global', 'waptservice_sslport'):
port = self.config.get('global', 'waptservice_sslport')
if port:
result['waptservice_protocol'] = 'https'
result['waptservice_port'] = int(port)
else:
result['waptservice_protocol'] = None
result['waptservice_port'] = None
elif self.config.has_option('global', 'waptservice_port'):
port = self.config.get('global', 'waptservice_port')
if port:
result['waptservice_protocol'] = 'http'
result['waptservice_port'] = int(port)
else:
# could be better
result['waptservice_protocol'] = None
result['waptservice_port'] = None
else:
# could be better
result['waptservice_protocol'] = 'http'
result['waptservice_port'] = 8088
result['repositories'] = [r.as_dict() for r in self.repositories]
if self.waptserver:
result['waptserver'] = self.waptserver.as_dict()
result['packages_whitelist'] = self.packages_whitelist
result['packages_blacklist'] = self.packages_blacklist
result['is_enterprise'] = self.is_enterprise()
#if self.is_enterprise():
# result['self_service_rules'] = self_service_rules(self)
return result
[docs] def reachable_ip(self):
"""Return the local IP which is most probably reachable by wapt server
In case there are several network connections, returns the local IP
which Windows choose for sending packets to WaptServer.
This can be the most probable IP which would get packets from WaptServer.
Returns:
str: Local IP
"""
try:
if self.waptserver and self.waptserver.server_url:
host = urllib.parse.urlparse(self.waptserver.server_url).hostname
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.settimeout(1)
s.connect((host, 0))
local_ip = s.getsockname()[0]
s.close()
return local_ip
else:
return None
except:
return None
[docs] def inventory(self):
"""Return full inventory of the computer as a dictionary.
Returns:
dict: {'host_info','wapt_status','installed_softwares','installed_packages'}
...changed:
1.4.1: renamed keys
1.6.2.4: removed setup.py from packages inventory.
"""
inv = {}
inv['host_info'] = setuphelpers.host_info()
inv['host_info']['repositories'] = ";".join([r.as_dict()['repo_url'] for r in self.repositories if not(r.as_dict()['repo_url'].endswith('-host'))])
# optionally forced dn
inv['computer_ad_dn'] = self.host_dn
inv['wapt_status'] = self.wapt_status()
inv['installed_softwares'] = self.merge_installed_softwares_and_wua_list()
inv['installed_packages'] = [p.as_dict() for p in self.waptdb.installed(include_errors=True, include_setup=False)]
inv['host_capabilities'] = self.host_capabilities()
return inv
[docs] def registration_inventory(self):
"""Minimum inventory for initial registration
"""
inv = {}
inv['computer_fqdn'] = ensure_unicode(setuphelpers.get_hostname())
inv['computer_ad_dn'] = self.host_dn
inv['wapt_status'] = self.wapt_status()
inv['host_capabilities'] = self.host_capabilities()
return inv
[docs] def personal_certificate(self):
"""Returns the personal certificates chain
Returns:
list (of SSLCertificate). The first one is the personal certificate. The other are useful if intermediate CA are used.
"""
cert_chain = SSLCABundle()
cert_chain.add_certificates_from_pem(pem_filename=self.personal_certificate_path)
return cert_chain.certificates()
[docs] def private_key(self, private_key_password=None):
"""SSLPrivateKey matching the personal_certificate
When key has been found, it is kept in memory for later use.
Args:
private_key_password : password to use to decrypt key. If None, passwd_callback is called.
Returns:
SSLPrivateKey
Raises:
EWaptMissingPrivateKey if ket can not be decrypted or found.
"""
if private_key_password is None:
password_callback = self.private_key_password_callback
else:
password_callback = None
certs = self.personal_certificate()
try:
cert = certs[0]
except IndexError:
raise EWaptMissingPrivateKey('No personal certificate could be found ; cannot look for any keys')
if not self._private_key_cache or not cert.match_key(self._private_key_cache):
self._private_key_cache = cert.matching_key_in_dirs(password_callback=password_callback, private_key_password=private_key_password)
if self._private_key_cache is None:
raise EWaptMissingPrivateKey('The key matching the certificate %s can not be found or decrypted' % (cert.public_cert_filename or cert.subject))
return self._private_key_cache
[docs] def cleanup_session_setup(self) -> int:
"""Remove all current user session_setup informations for removed packages
"""
# run session_cleanup for all uninstalled packages
result = {'done':[],'errors':[]}
session_setup_package_uuids = [ p[0] for p in self.waptsessiondb.query("select package_uuid from wapt_sessionsetup where install_status='OK' order by install_date", as_dict=False) ]
for package_uuid in session_setup_package_uuids:
# uninstalled package if no other package with same name is installed
pe_dict = self.waptpublicdb.query("""select * from wapt_publicstatus s where s.install_date is not null and s.uninstall_date is not null and s.package_uuid=?
and not exists (select i.package_uuid from wapt_publicstatus i where i.package=s.package and i.install_date is not null and i.uninstall_date is null) order by s.uninstall_date""",[package_uuid],one=True)
if pe_dict:
has_session_cleanup = 'def session_cleanup():' in pe_dict.get('setuppy','')
if has_session_cleanup:
pe = PackageEntry(**pe_dict)
pe.setuppy = pe_dict.get('setuppy')
pe._control_lines = pe_dict.get('control','').splitlines()
try:
print(pe.call_setup_hook('session_cleanup',wapt_context=self, user=self.user))
self.waptsessiondb.execute("""delete from wapt_sessionsetup where package_uuid=?""",[package_uuid])
result['done'].append(package_uuid)
except Exception as e:
logger.critical('Unable to run session_cleanup for package %s and user %s: %s' % (pe.package, self.user, e))
result['errors'].append(package_uuid)
installed = [p['package'] for p in self.waptpublicdb.query("select distinct package from wapt_publicstatus where install_date is not null and uninstall_date is null")]
return self.waptsessiondb.remove_obsolete_install_status(installed)
[docs] def session_setup(self, package, force=False, params=None):
"""Setup the user session for a specific system wide installed package"
Source setup.py from database or filename
"""
install_id = None
oldpath = sys.path
try:
is_dev_mode = False
package_entry = None
if isinstance(package, PackageEntry):
package_entry = package
elif os.path.isdir(package):
package_entry = PackageEntry().load_control_from_wapt(package)
is_dev_mode = True
else:
public_status = self.waptpublicdb.query('select * from wapt_publicstatus where package=? and install_date is not null and uninstall_date is null order by install_date desc',
[package],one=True)
if public_status:
package_entry = PackageEntry()
package_entry._load_control(public_status['control'])
package_entry['setuppy'] = public_status['setuppy']
if not package_entry:
raise Exception('Package %s is not installed' % package)
if package_entry.has_setup_py():
if (is_dev_mode or 'def session_setup():' in package_entry.setuppy):
# initialize a session db for the user
session_db = WaptSessionDB(self.user) # WaptSessionDB()
with session_db:
if force or is_dev_mode or not session_db.is_installed(package_entry.package, package_entry.version):
print(("Running session_setup for package %s and user %s" % (package_entry.asrequirement(), self.user)))
install_id = session_db.add_start_install(package_entry)
with WaptPackageSessionSetupLogger(console=sys.stderr, waptsessiondb=session_db, install_id=install_id) as dblog:
try:
# get value of required parameters from system wide install
try:
result = package_entry.call_setup_hook('session_setup', self, force=force, params=params)
except EWaptMissingPackageHook:
result = None
if result:
dblog.exit_status = 'RETRY'
session_db.update_install_status(install_id, append_line='session_setup() done\n')
else:
dblog.exit_status = 'OK'
session_db.update_install_status(install_id, append_line='session_setup() done\n')
return result
except Exception:
logger.critical("session_setup failed for package %s and user %s" % (package_entry.asrequirement(), self.user))
session_db.update_install_status(install_id, append_line=traceback.format_exc())
dblog.exit_status = 'ERROR'
else:
logger.info("session_setup for package %s and user %s already installed" % (package_entry.asrequirement(), self.user))
else:
logger.debug('No session_setup for package %s' % package_entry['package'])
else:
logger.debug('No setup.py for %s, skipping session-setup' % package_entry['package'])
finally:
sys.path = oldpath
[docs] def audit(self, package, force=False, ignore_schedule=False, audited_by=None) -> str:
"""Run the audit hook for the installed package"
Source setup.py from database, filename, or packageEntry
Stores the result and log into "wapt_localstatus" table
Args:
package (PackageEntry or directory or package name or {package_uuid} ): package to audit
Returns:
str : iso datetime of this audit as stored in packages status table
"""
def worst(r1, r2):
states = ['OK', 'WARNING', 'ERROR', 'UNKNOWN']
try:
idxr1 = states.index(r1)
except ValueError:
idxr1 = states.index('UNKNOWN')
try:
idxr2 = states.index(r2)
except ValueError:
idxr2 = states.index('UNKNOWN')
if idxr1 > idxr2:
return states[idxr1]
else:
return states[idxr2]
install_id = None
now = datetime2isodate(datetime.datetime.utcnow())
oldpath = sys.path
try:
if isinstance(package, PackageEntry):
package_entry = package
elif os.path.isdir(package):
package_entry = PackageEntry().load_control_from_wapt(package)
else:
package_entry = self.is_installed(package)
if not package_entry:
raise Exception('Package %s is not installed' % package)
if hasattr(package_entry, 'install_status') and hasattr(package_entry, 'rowid'):
install_id = package_entry.rowid
package_install = package_entry
else:
install_id = self._get_package_status_rowid(package_entry)
if install_id is None:
raise Exception('Package %s is not installed' % package)
package_install = self.waptdb.install_status(install_id)
if ignore_schedule or force or not package_install.next_audit_on or now >= package_install.next_audit_on:
next_audit = None
if package_install.audit_schedule:
# audit_schedule can contains several items like 1d,on_connect
audit_schedule = ensure_list(package_install.audit_schedule,allow_none=True)
else:
audit_schedule = ensure_list(self.waptaudit_task_period,allow_none=True)
if audit_schedule is not None:
for schedule in audit_schedule:
timedelta = get_time_delta(schedule, 'm')
if timedelta:
# we use the earliest next audit
if next_audit is None or datetime.datetime.utcnow() + timedelta < next_audit:
next_audit = datetime.datetime.utcnow() + timedelta
# skip audit entirely if no uninstall_key and no audit hook
if not package_install['uninstall_key'] and (not package_entry.has_setup_py() or not 'def audit():' in package_entry.setuppy):
self.waptdb.update_audit_status(install_id, set_status='OK', set_last_audit_on=datetime2isodate(datetime.datetime.utcnow()), set_next_audit_on=next_audit and datetime2isodate(next_audit) or None)
return 'OK'
logger.info("Audit run for package %s and user %s" % (package, audited_by or self.user))
self.waptdb.update_audit_status(install_id, set_status='RUNNING', set_output='',
set_last_audit_on=datetime2isodate(datetime.datetime.utcnow()),
set_next_audit_on=next_audit and datetime2isodate(next_audit) or None)
with WaptPackageAuditLogger(console=sys.stderr, wapt_context=self, install_id=install_id, user=audited_by or self.user) as dblog:
try:
# check if registered uninstalley are still there
uninstallkeys = self._get_uninstallkeylist(package_install['uninstall_key'])
dblog.exit_status = 'OK'
print('Auditing %s' % package_entry.package)
if sys.platform == 'win32':
if uninstallkeys is not None:
for key in uninstallkeys:
uninstallkey_exists = setuphelpers.installed_softwares(uninstallkey=key)
if not uninstallkey_exists:
print('ERROR: Uninstall Key %s is not in Windows Registry.' % key)
dblog.exit_status = worst(dblog.exit_status, 'ERROR')
else:
print(' OK: Uninstall Key %s in Windows Registry.' % key)
dblog.exit_status = worst(dblog.exit_status, 'OK')
elif sys.platform == 'darwin':
if uninstallkeys is not None:
for key in uninstallkeys:
if not setuphelpers.uninstall_key_exists(key):
print('ERROR: Uninstall Key %s is not installed.' % key)
dblog.exit_status = worst(dblog.exit_status, 'ERROR')
else:
print(' OK: Uninstall Key %s.' % key)
dblog.exit_status = worst(dblog.exit_status, 'OK')
if package_entry.has_setup_py():
# get value of required parameters from system wide install
params = self.get_previous_package_params(package_entry)
# this call return None if not audit hook or if hook has no return value.
try:
result = package_entry.call_setup_hook('audit', self, params,force=force, user=audited_by)
except EWaptMissingPackageHook:
result = 'OK'
dblog.exit_status = worst(dblog.exit_status, result)
else:
logger.debug('No setup.py, skipping session-setup')
print( 'OK: No setup.py')
dblog.exit_status = worst(dblog.exit_status, 'OK')
return dblog.exit_status
except Exception as e:
print('Audit aborted due to exception: %s' % e)
dblog.exit_status = 'ERROR'
return dblog.exit_status
else:
return package_install.last_audit_status
finally:
sys.path = oldpath
[docs] def get_previous_package_params(self, package_entry):
"""Return the params used when previous install of package_entry.package
If no previous install, return {}
The params are stored as json string in local package status table.
Args:
package_entry (PackageEntry): package request to lookup.
Returns:
dict
"""
# get old install params if the package has been already installed
old_install = self.is_installed(package_entry.package)
if old_install:
return ujson.loads(old_install['install_params'])
else:
return {}
[docs] def uninstall(self, packagename, params_dict={},force=False):
"""Launch the uninstall script of an installed package"
Source setup.py from database or filename
"""
try:
previous_cwd = os.getcwd()
if os.path.isdir(packagename):
entry = PackageEntry().load_control_from_wapt(packagename)
else:
logger.debug('Sourcing setup from DB')
entry = self.is_installed(packagename)
if not entry:
raise Exception('no package %s installed on this host' % packagename)
params = self.get_previous_package_params(entry)
params.update(params_dict)
if entry.has_setup_py():
try:
entry.call_setup_hook('uninstall', self, params=params,force=force)
except EWaptMissingPackageHook:
pass
else:
logger.info('Uninstall: no setup.py source in database.')
finally:
logger.debug(' Change current directory to %s' % previous_cwd)
os.chdir(previous_cwd)
def __get_installer_defaults_deb(self, result_format, installer_path):
"""See get_installer_defaults(). Specific to .deb archives"""
result = result_format
try:
with arpy.Archive(installer_path) as ar_file:
tarball = ar_file.open('control.tar.gz')
tar_file = tarfile.open(fileobj=tarball)
control_file = tar_file.extractfile('./control')
control_lines = control_file.readlines()
control_dict = {}
for line in control_lines:
line_split = line.decode('utf-8').split(': ')
control_dict[line_split[0]] = line_split[1]
result.update(dict(filename=control_dict['Package'],
version=control_dict['Version'],
description=control_dict['Description'],
architecture=control_dict['Architecture']))
except Exception as e:
logger.info('Couldn\'t extract metadata from deb archive {} : {}'.format(installer_path, e))
return result
def __get_installer_defaults_rpm(self, result_format, installer_path):
"""See get_installer_defaults(). Specific to .deb archives"""
result = result_format
try:
with rpmfile.open(installer_path) as rpm:
rpm_headers = rpm.headers
result.update(dict(filename=rpm_headers['name'],
version=rpm_headers['version'],
description=rpm_headers['description'],
architecture=rpm_headers['arch']))
except Exception as e:
logger.info('Couldn\'t extract metadata from rpm archive {} : {}'.format(installer_path, e))
return result
[docs] def get_installer_defaults(self, installer_path)->Dict:
"""Returns guessed default values for package templates based on installer binary
Args:
installer_path (str): filepath to installer
Returns:
dict:
>>> get_installer_defaults(r'c:\tranquilit\wapt\tests\SumatraPDF-3.1.1-install.exe')
{'description': u'SumatraPDF Installer (Krzysztof Kowalczyk)',
'filename': 'SumatraPDF-3.1.1-install.exe',
'silentflags': '/VERYSILENT',
'simplename': u'sumatrapdf-installer',
'type': 'UnknownExeInstaller',
'version': u'3.1.1'}
>>> get_installer_defaults(r'c:\tranquilit\wapt\tests\7z920.msi')
{'description': u'7-Zip 9.20 (Igor Pavlov)',
'filename': '7z920.msi',
'silentflags': '/q /norestart',
'simplename': u'7-zip-9.20',
'type': 'MSI',
'version': u'9.20.00.0'}
"""
(product_name, ext) = os.path.splitext(installer_path)
ext = ext.lower()
props = setuphelpers.get_product_props(installer_path)
simplename = re.sub(r'[\s\(\)]+', '-', props['product'].lower())
description = props['description']
publisher = props['publisher']
version = props['version'] or '0.0.0'
result = dict(filename=os.path.basename(installer_path),
simplename=simplename,
version=version,
description=description,
silentflags='',
type=None,
uninstallkey=None,
publisher=publisher,
architecture='all')
# the get_installer_defaults_win function is only in setuphelpers_windows
if sys.platform == 'win32' and (ext in ['.exe', '.msi', '.msu']):
return setuphelpers.get_installer_defaults_win(installer_path)
if ext == '.deb':
result = self.__get_installer_defaults_deb(result, installer_path)
elif ext == '.rpm':
result = self.__get_installer_defaults_rpm(result, installer_path)
else:
result.update(dict(type=setuphelpers.InstallerTypes.UnknownInstaller, silentflags='/VERYSILENT'))
return result
[docs] def make_package_template(self, installer_path='', packagename='', directoryname='',
section='base', description=None, depends='', version=None, silentflags=None, uninstallkey=None,
maturity=None, architecture='all', target_os='all', product_name=None)->str:
r"""Build a skeleton of WAPT package based on the properties of the supplied installer
Return the path of the skeleton
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> wapt.dbpath = ':memory:'
>>> files = 'c:/tmp/files'
>>> if not os.path.isdir(files):
... os.makedirs(files)
>>> tmpdir = 'c:/tmp/dummy'
>>> devdir = wapt.make_package_template(files,packagename='mydummy',directoryname=tmpdir,depends='tis-firefox')
>>> os.path.isfile(os.path.join(devdir,'WAPT','control'))
True
>>> p = wapt.build_package(devdir)
>>> 'filename' in p and isinstance(p['files'],list) and isinstance(p['package'],PackageEntry)
True
>>> import shutil
>>> shutil.rmtree(tmpdir)
"""
if installer_path:
installer_path = os.path.abspath(installer_path)
if directoryname:
directoryname = os.path.abspath(directoryname)
if not installer_path and not packagename:
raise EWaptException('You must provide at least installer_path or packagename to be able to prepare a package template')
if installer_path:
installer = os.path.basename(installer_path)
else:
installer = ''
uninstallkey = uninstallkey or ''
product_name = product_name or ''
if os.path.isfile(installer_path):
# case of an installer
props = setuphelpers.get_product_props(installer_path)
silentflags = silentflags or setuphelpers.getsilentflags(installer_path)
# for MSI, uninstallkey is in properties
if not uninstallkey and 'ProductCode' in props:
uninstallkey = '"%s"' % props['ProductCode']
elif os.path.isdir(installer_path):
# case of a directory
props = {
'product': installer,
'description': installer,
'version': '0',
'publisher': ensure_unicode(setuphelpers.get_current_user()),
'ProductName' : product_name
}
silentflags = silentflags or ''
else:
# case of a nothing
props = {
'product': packagename,
'description': packagename,
'version': '0',
'publisher': ensure_unicode(setuphelpers.get_current_user()),
'ProductName' : product_name
}
silentflags = ''
if not packagename:
simplename = re.sub(r'[\s\(\)\|\,\.\%]+', '_', props['product'].lower())
packagename = '%s-%s' % (self.config.get('global', 'default_package_prefix'), simplename)
description = description or 'Package for %s ' % props['description']
version = version or props['version']
product_name = product_name or props['ProductName']
if not directoryname:
directoryname = self.get_default_development_dir(PackageEntry(package=packagename, version=version, section=section, maturity=maturity, target_os=target_os, architecture=architecture))
if not os.path.isdir(os.path.join(directoryname, 'WAPT')):
os.makedirs(os.path.join(directoryname, 'WAPT'))
if installer_path:
(installer_name, installer_ext) = os.path.splitext(installer)
installer_ext = installer_ext.lower()
if installer_ext in ['.msi', '.msix']:
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_msi.py.tmpl')
elif installer_ext == '.msu':
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_msu.py.tmpl')
elif installer_ext == '.exe':
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_exe.py.tmpl')
elif installer_ext == '.deb':
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_deb.py.tmpl')
elif installer_ext == '.pkg':
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_pkg.py.tmpl')
elif installer_ext == '.dmg':
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_dmg.py.tmpl')
elif installer_ext == '.rpm':
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_rpm.py.tmpl')
elif installer_ext == '.crt':
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_cert.py.tmpl')
elif os.path.isdir(installer_path):
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template_dir.py.tmpl')
else:
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_template.py.tmpl')
else:
setup_template = os.path.join(self.wapt_base_dir, 'templates', 'setup_package_skel.py.tmpl')
template = codecs.open(setup_template, encoding='utf8').read() % dict(
packagename=packagename,
uninstallkey=uninstallkey,
silentflags=silentflags,
installer=installer,
product=props['product'],
description=description,
version=version,
)
setuppy_filename = os.path.join(directoryname, 'setup.py')
if not os.path.isfile(setuppy_filename):
codecs.open(setuppy_filename, 'w', encoding='utf8').write(template)
else:
logger.info('setup.py file already exists, skip create')
logger.debug('Copy installer %s to target' % installer)
if os.path.isfile(installer_path):
shutil.copyfile(installer_path, os.path.join(directoryname, installer))
elif os.path.isdir(installer_path):
setuphelpers.copytree2(installer_path, os.path.join(directoryname, installer))
control_filename = os.path.join(directoryname, 'WAPT', 'control')
if not os.path.isfile(control_filename):
entry = PackageEntry()
entry.package = packagename
entry.name = product_name
entry.architecture = architecture
entry.target_os = target_os
if maturity is None:
entry.maturity = self.default_maturity
else:
entry.maturity = maturity
entry.description = description
try:
entry.maintainer = ensure_unicode(win32api.GetUserNameEx(3))
except:
try:
entry.maintainer = ensure_unicode(setuphelpers.get_current_user())
except:
entry.maintainer = os.environ['USERNAME']
entry.priority = 'optional'
entry.section = section or 'base'
entry.version = version+'-0'
entry.depends = depends
if self.config.has_option('global', 'default_sources_url'):
entry.sources = self.config.get('global', 'default_sources_url') % entry.as_dict()
codecs.open(control_filename, 'w', encoding='utf8').write(entry.ascontrol())
else:
logger.info('control file already exists, skip create')
self.add_pyscripter_project(directoryname)
self.add_vscode_project(directoryname)
return directoryname
[docs] def make_host_template(self, packagename='', depends=None, conflicts=None, directoryname=None, description=None)->PackageEntry:
if not packagename:
packagename = self.host_packagename()
return self.make_group_template(packagename=packagename, depends=depends, conflicts=conflicts, directoryname=directoryname, section='host', description=description)
[docs] def make_group_template(self, packagename='', maturity=None, depends=None, conflicts=None, directoryname=None, section='group', description=None)->PackageEntry:
r"""Creates or updates on disk a skeleton of a WAPT group package.
If the a package skeleton already exists in directoryname, it is updated.
sourcespath attribute of returned PackageEntry is populated with the developement directory of group package.
Args:
packagename (str): group name
depends :
conflicts
directoryname
section
description
Returns:
PackageEntry
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> tmpdir = 'c:/tmp/dummy'
>>> if os.path.isdir(tmpdir):
... import shutil
... shutil.rmtree(tmpdir)
>>> p = wapt.make_group_template(packagename='testgroupe',directoryname=tmpdir,depends='tis-firefox',description=u'Test de groupe')
>>> print p
>>> print p['package'].depends
tis-firefox
>>> import shutil
>>> shutil.rmtree(tmpdir)
"""
if directoryname:
directoryname = os.path.abspath(directoryname)
if not packagename:
packagename = self.host_packagename()
if not directoryname:
directoryname = self.get_default_development_dir(packagename, section=section)
if not directoryname:
directoryname = tempfile.mkdtemp('wapt')
if not os.path.isdir(os.path.join(directoryname, 'WAPT')):
os.makedirs(os.path.join(directoryname, 'WAPT'))
template_fn = os.path.join(self.wapt_base_dir, 'templates', 'setup_%s_template.py' % section)
if os.path.isfile(template_fn):
# replacing %(var)s by local values in template
# so setup template must use other string formating system than % like '{}'.format()
template = codecs.open(template_fn, encoding='utf8').read() % locals()
setuppy_filename = os.path.join(directoryname, 'setup.py')
if not os.path.isfile(setuppy_filename):
codecs.open(setuppy_filename, 'w', encoding='utf8').write(template)
else:
logger.info('setup.py file already exists, skip create')
else:
logger.info('No %s template. Package wil lhave no setup.py' % template_fn)
control_filename = os.path.join(directoryname, 'WAPT', 'control')
entry = PackageEntry()
if not os.path.isfile(control_filename):
entry.priority = 'standard'
entry.section = section
entry.version = '0'
entry.architecture = 'all'
if maturity is None:
entry.maturity = maturity
else:
entry.maturity = self.default_maturity
entry.description = description or '%s package for %s ' % (section, packagename)
try:
entry.maintainer = ensure_unicode(win32api.GetUserNameEx(3))
except:
try:
entry.maintainer = ensure_unicode(setuphelpers.get_current_user())
except:
entry.maintainer = os.environ['USERNAME']
else:
entry.load_control_from_wapt(directoryname)
entry.package = packagename
# Check existing versions and increment it
older_packages = self.is_available(entry.package)
if older_packages and entry <= older_packages[-1]:
entry.version = older_packages[-1].version
entry.inc_build()
entry.filename = entry.make_package_filename()
if self.config.has_option('global', 'default_sources_url'):
entry.sources = self.config.get('global', 'default_sources_url') % {'packagename': packagename}
# check if depends should be appended to existing depends
if (isinstance(depends, str) or isinstance(depends, str)) and depends.startswith('+'):
append_depends = True
depends = ensure_list(depends[1:])
current = ensure_list(entry.depends)
for d in depends:
if not d in current:
current.append(d)
depends = current
else:
append_depends = False
depends = ensure_list(depends)
if depends:
# use supplied list of packages
entry.depends = ','.join(['%s' % p for p in depends if p and p != packagename])
# check if conflicts should be appended to existing conflicts
if (isinstance(conflicts, str) or isinstance(conflicts, str)) and conflicts.startswith('+'):
append_conflicts = True
conflicts = ensure_list(conflicts[1:])
current = ensure_list(entry.conflicts)
for d in conflicts:
if not d in current:
current.append(d)
conflicts = current
else:
append_conflicts = False
conflicts = ensure_list(conflicts)
if conflicts:
# use supplied list of packages
entry.conflicts = ','.join(['%s' % p for p in conflicts if p and p != packagename])
entry.save_control_to_wapt(directoryname)
if entry.section != 'host':
self.add_pyscripter_project(directoryname)
self.add_vscode_project(directoryname)
return entry
[docs] def is_installed(self, packagename: str, include_errors=False) -> PackageEntry:
"""Checks if a package is installed.
Return package entry and additional local status or None
Args:
packagename (str): name / {package_uuid}/ package request to query
Returns:
PackageEntry: None or PackageEntry merged with local install_xxx fields
* install_date
* install_output
* install_params
* install_status
"""
if isinstance(packagename, PackageEntry):
packagename = packagename.asrequirement()
package_entry = self.waptdb.installed_matching(packagename, include_errors=include_errors, include_setup=True, include_control=True)
return package_entry
[docs] def installed(self, include_errors=False)->List[PackageEntry]:
"""Returns all installed packages with their status
Args:
include_errors (boolean): include packages wnot installed successfully
Returns:
list: list of PackageEntry merged with local install status.
"""
return self.waptdb.installed(include_errors=include_errors)
[docs] def is_available(self, packagename)->List[PackageEntry]:
r"""Check if a package (with optional version condition) is available
in repositories.
Args:
packagename (str) : package name to lookup or package requirement ( packagename(=version) )
Returns:
list : of PackageEntry sorted by package version ascending
>>> wapt = Wapt(config_filename='c:/tranquilit/wapt/tests/wapt-get.ini')
>>> l = wapt.is_available('tis-wapttest')
>>> l and isinstance(l[0],PackageEntry)
True
"""
return self.waptdb.packages_matching(packagename)
[docs] def get_default_development_dir(self, packagecond:str, section='base')->str:
"""Returns the default development directory for package named <packagecond>
based on default_sources_root ini parameter if provided
Args:
packagecond (PackageEntry or str): either PackageEntry or a "name(=version)" string
Returns:
unicode: path to local proposed development directory
"""
if not isinstance(packagecond, PackageEntry):
# assume something like "package(=version)"
package_and_version = REGEX_PACKAGE_CONDITION.match(packagecond).groupdict()
pe = PackageEntry(package_and_version['package'], package_and_version['version'] or '0')
else:
pe = packagecond
root = ensure_unicode(self.config.get('global', 'default_sources_root'))
if not root:
root = ensure_unicode(tempfile.gettempdir())
return os.path.join(root, pe.make_package_edit_directory())
[docs] def add_pyscripter_project(self, target_directory):
"""Add a pyscripter project file to package development directory.
Args:
target_directory (str): path to location where to create the wapt.psproj file.
Returns:
None
"""
psproj_filename = os.path.join(target_directory, 'WAPT', 'wapt.psproj')
# if not os.path.isfile(psproj_filename):
# supply some variables to psproj template
datas = self.as_dict()
datas['target_directory'] = target_directory
proj_template = codecs.open(os.path.join(self.wapt_base_dir, 'templates', 'wapt.psproj'), encoding='utf8').read() % datas
codecs.open(psproj_filename, 'w', encoding='utf8').write(proj_template)
[docs] def add_vscode_project(self, target_directory):
r"""Add .vscode folder with project files to the package development directory
Args:
target_directory (str): path to the package development directory.
Returns:
None
"""
vscode_dir = os.path.join(target_directory, ".vscode")
if not (os.path.isdir(vscode_dir)):
os.mkdir(vscode_dir)
launch_json = os.path.join(vscode_dir, "launch.json")
if os.path.isfile(launch_json):
os.remove(launch_json)
shutil.copyfile(os.path.join(self.wapt_base_dir, "templates", "vscode_launch.json"), launch_json)
with open(os.path.join(self.wapt_base_dir, "templates", "vscode_settings.json"), "r") as settings_json_file:
settings_json = json.load(settings_json_file)
if not (os.path.isfile(os.path.join(self.wapt_base_dir, "python.exe"))):
settings_json["python.pythonPath"] = os.path.join(self.wapt_base_dir, "bin", "python")
settings_json["python.defaultInterpreterPath"] = os.path.join(self.wapt_base_dir, "bin", "python")
else:
settings_json["python.pythonPath"] = os.path.join(self.wapt_base_dir, "python.exe")
settings_json["python.defaultInterpreterPath"] = os.path.join(self.wapt_base_dir, "python.exe")
settings_json["python.wapt-get"] = os.path.join(self.wapt_base_dir, "wapt-get.py")
with open(os.path.join(vscode_dir, "settings.json"), "w") as settings_json_outfile:
json.dump(settings_json, settings_json_outfile, indent=4)
with open(os.path.join(target_directory, ".env"), "w") as fenv:
list_of_env = ["VIRTUAL_ENV=" + self.wapt_base_dir, "PYTHONPATH=" + self.wapt_base_dir]
fenv.write("\n".join(list_of_env) + "\n")
[docs] def edit_package(self, packagerequest,
target_directory='',
use_local_sources=True,
append_depends=None,
remove_depends=None,
append_conflicts=None,
remove_conflicts=None,
auto_inc_version=True,
cabundle=None,
):
r"""Download an existing package from repositories into target_directory for modification
if use_local_sources is True and no newer package exists on repos, updates current local edited data
else if target_directory exists and is not empty, raise an exception
Args:
packagerequest (str) : path to existing wapt file, or package request
use_local_sources (boolean) : don't raise an exception if target exist and match package version
append_depends (list of str) : package requirements to add to depends
remove_depends (list or str) : package requirements to remove from depends
auto_inc_version (bool) :
cabundle (SSLCABundle) : list of authorized certificate filenames. If None, use default from current wapt.
Returns:
PackageEntry : edit local package with sourcespath attribute populated
>>> wapt = Wapt(config_filename='c:/tranquilit/wapt/tests/wapt-get.ini')
>>> wapt.dbpath = ':memory:'
>>> r= wapt.update()
>>> tmpdir = tempfile.mkdtemp('wapt')
>>> res = wapt.edit_package('tis-wapttest',target_directory=tmpdir,append_depends='tis-firefox',remove_depends='tis-7zip')
>>> res['target'] == tmpdir and res['package'].package == 'tis-wapttest' and 'tis-firefox' in res['package'].depends
True
>>> import shutil
>>> shutil.rmtree(tmpdir)
"""
if cabundle is None:
cabundle = self.cabundle
# check before if path exist
if os.path.isdir(packagerequest):
entry = PackageEntry(waptfile=packagerequest)
entry.localpath=packagerequest
target_directory=packagerequest
elif os.path.isfile(packagerequest):
entry = PackageEntry(waptfile=packagerequest)
else:
# check if available in repos
entries = self.is_available(packagerequest)
if entries:
entry = entries[-1]
self.download_packages(entry)
else:
raise EWaptException('Package %s does not exist. Either update local status or check filepath.' % (packagerequest))
packagerequest = entry.asrequirement()
if target_directory is None:
target_directory = tempfile.mkdtemp(prefix="wapt")
elif not target_directory:
target_directory = self.get_default_development_dir(entry, section=entry.section)
if entry.localpath:
local_dev_entry = self.is_wapt_package_development_dir(target_directory)
if local_dev_entry:
if use_local_sources and not local_dev_entry.match(packagerequest):
raise Exception('Target directory %s contains a different package version %s' % (target_directory, entry.asrequirement()))
elif not use_local_sources:
raise Exception('Target directory %s contains already a developement package %s' % (target_directory, entry.asrequirement()))
else:
logger.info('Using existing development sources %s' % target_directory)
elif not local_dev_entry:
entry.unzip_package(target_dir=target_directory, cabundle=cabundle)
entry.invalidate_signature()
local_dev_entry = entry
append_depends = ensure_list(append_depends)
remove_depends = ensure_list(remove_depends)
append_conflicts = ensure_list(append_conflicts)
remove_conflicts = ensure_list(remove_conflicts)
if append_depends or remove_depends or append_conflicts or remove_conflicts:
prev_depends = ensure_list(local_dev_entry.depends)
for d in append_depends:
if not d in prev_depends:
prev_depends.append(d)
for d in remove_depends:
if d in prev_depends:
prev_depends.remove(d)
prev_conflicts = ensure_list(local_dev_entry.conflicts)
for d in append_conflicts:
if not d in prev_conflicts:
prev_conflicts.append(d)
if remove_conflicts:
for d in remove_conflicts:
if d in prev_conflicts:
prev_conflicts.remove(d)
local_dev_entry.depends = ','.join(prev_depends)
local_dev_entry.conflicts = ','.join(prev_conflicts)
local_dev_entry.save_control_to_wapt(target_directory)
if entry.section != 'host':
self.add_pyscripter_project(target_directory)
self.add_vscode_project(target_directory)
return local_dev_entry
else:
raise Exception('Unable to unzip package in %s' % target_directory)
[docs] def is_wapt_package_development_dir(self, directory):
"""Return PackageEntry if directory is a wapt developement directory (a WAPT/control file exists) or False"""
return os.path.isfile(os.path.join(directory, 'WAPT', 'control')) and PackageEntry().load_control_from_wapt(directory, calc_md5=False)
[docs] def is_wapt_package_file(self, filename):
"""Return PackageEntry if filename is a wapt package or False
True if file ends with .wapt and control file can be loaded and decoded from zip file
Args:
filename (str): path to a file
Returns:
False or PackageEntry
"""
(root, ext) = os.path.splitext(filename)
if ext != '.wapt' or not os.path.isfile(filename):
return False
try:
entry = PackageEntry().load_control_from_wapt(filename, calc_md5=False)
return entry
except:
return False
[docs] def edit_host(self,
hostname,
target_directory=None,
append_depends=None,
remove_depends=None,
append_conflicts=None,
remove_conflicts=None,
printhook=None,
description=None,
cabundle=None,
):
"""Download and extract a host package from host repositories into target_directory for modification
Args:
hostname (str) : fqdn of the host to edit
target_directory (str) : where to place the developments files. if empty, use default one from wapt-get.ini configuration
append_depends (str or list) : list or comma separated list of package requirements
remove_depends (str or list) : list or comma separated list of package requirements to remove
cabundle (SSLCA Bundle) : authorized ca certificates. If None, use default from current wapt.
Returns:
PackageEntry
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> tmpdir = 'c:/tmp/dummy'
>>> wapt.edit_host('dummy.tranquilit.local',target_directory=tmpdir,append_depends='tis-firefox')
>>> import shutil
>>> shutil.rmtree(tmpdir)
>>> host = wapt.edit_host('htlaptop.tranquilit.local',target_directory=tmpdir,append_depends='tis-firefox')
>>> 'package' in host
True
>>> shutil.rmtree(tmpdir)
"""
if target_directory is None:
target_directory = tempfile.mkdtemp('wapt')
elif not target_directory:
target_directory = self.get_default_development_dir(hostname, section='host')
if os.path.isdir(target_directory) and os.listdir(target_directory):
raise Exception('directory %s is not empty, aborting.' % target_directory)
#self.use_hostpackages = True
if cabundle is None:
cabundle = self.cabundle
append_depends = ensure_list(append_depends)
remove_depends = ensure_list(remove_depends)
append_conflicts = ensure_list(append_conflicts)
remove_conflicts = ensure_list(remove_conflicts)
for d in append_depends:
if not d in remove_conflicts:
remove_conflicts.append(d)
for d in append_conflicts:
if not d in remove_depends:
remove_depends.append(d)
# create a temporary repo for this host
host_repo = WaptHostRepo(name='wapt-host', host_id=hostname, config=self.config, host_key=self._host_key, WAPT=self)
entry = host_repo.get(hostname)
if entry:
host_repo.download_packages(entry)
entry.unzip_package(target_dir=target_directory, cabundle=cabundle)
entry.invalidate_signature()
# update depends list
prev_depends = ensure_list(entry.depends)
for d in append_depends:
if not d in prev_depends:
prev_depends.append(d)
for d in remove_depends:
if d in prev_depends:
prev_depends.remove(d)
entry.depends = ','.join(prev_depends)
# update conflicts list
prev_conflicts = ensure_list(entry.conflicts)
for d in append_conflicts:
if not d in prev_conflicts:
prev_conflicts.append(d)
if remove_conflicts:
for d in remove_conflicts:
if d in prev_conflicts:
prev_conflicts.remove(d)
entry.conflicts = ','.join(prev_conflicts)
if description is not None:
entry.description = description
entry.save_control_to_wapt(target_directory)
return entry
else:
# create a new version of the existing package in repository
return self.make_host_template(packagename=hostname, directoryname=target_directory, depends=append_depends, description=description)
[docs] def forget_packages(self, packages_list):
"""Remove install status for packages from local database
without actually uninstalling the packages
Args:
packages_list (list): list of installed package names to forget
Returns:
list: list of package names actually forgotten
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> res = wapt.install('tis-test')
???
>>> res = wapt.is_installed('tis-test')
>>> isinstance(res,PackageEntry)
True
>>> wapt.forget_packages('tis-test')
['tis-test']
>>> wapt.is_installed('tis-test')
>>> print wapt.is_installed('tis-test')
None
"""
result = []
packages_list = ensure_list(packages_list)
for package in packages_list:
q = self.waptdb.query("""\
select * from wapt_localstatus
where package=?
""", (package,))
for pe in q:
self.remove_package_persistent_dirs(pe.get('package_uuid'))
rowid = self.waptdb.remove_install_status(package)
if rowid:
result.append(package)
self.store_upgrade_status()
return result
[docs] def duplicate_package(self,
packagename,
newname=None,
newversion=None,
newmaturity=None,
target_directory=None,
append_depends=None,
remove_depends=None,
append_conflicts=None,
remove_conflicts=None,
auto_inc_version=True,
usecache=True,
printhook=None,
cabundle=None,
):
"""Duplicate an existing package.
Duplicate an existing package from declared repostory or file into targetdirectory with
optional newname and version.
Args:
packagename (str) : packagename to duplicate, or filepath to a local package or package development directory.
newname (str): name of target package
newversion (str): version of target package. if None, use source package version
target_directory (str): path where to put development files. If None, use temporary. If empty, use default development dir
append_depends (list): comma str or list of depends to append.
remove_depends (list): comma str or list of depends to remove.
auto_inc_version (bool): if version is less than existing package in repo, set version to repo version+1
usecache (bool): If True, allow to use cached package in local repo instead of downloading it.
printhook (func): hook for download progress
cabundle (SSLCABundle): list of authorized ca certificate (SSLPublicCertificate) to check authenticity of source packages. If None, no check is performed.
Returns:
PackageEntry : new packageEntry with sourcespath = target_directory
>>> wapt = Wapt(config_filename='c:/tranquilit/wapt/tests/wapt-get.ini')
>>> wapt.dbpath = ':memory:'
>>> r= wapt.update()
>>> def nullhook(*args):
... pass
>>> tmpdir = 'c:/tmp/testdup-wapt'
>>> if os.path.isdir(tmpdir):
... import shutil
... shutil.rmtree(tmpdir)
>>> p = wapt.duplicate_package('tis-wapttest',
... newname='testdup',
... newversion='20.0-0',
... target_directory=tmpdir,
... excludes=['.svn','.git','.gitignore','*.pyc','src'],
... append_depends=None,
... auto_inc_version=True,
... usecache=False,
... printhook=nullhook)
>>> print repr(p['package'])
PackageEntry('testdup','20.0-0')
>>> if os.path.isdir(tmpdir):
... import shutil
... shutil.rmtree(tmpdir)
>>> p = wapt.duplicate_package('tis-wapttest',
... target_directory=tempfile.mkdtemp('wapt'),
... auto_inc_version=True,
... append_depends=['tis-firefox','tis-irfanview'],
... remove_depends=['tis-wapttestsub'],
... )
>>> print repr(p['package'])
PackageEntry('tis-wapttest','120')
"""
if target_directory:
target_directory = os.path.abspath(target_directory)
if newname:
while newname.endswith('.wapt'):
dot_wapt = newname.rfind('.wapt')
newname = newname[0:dot_wapt]
logger.warning("Target ends with '.wapt', stripping. New name: %s", newname)
append_depends = ensure_list(append_depends)
remove_depends = ensure_list(remove_depends)
append_conflicts = ensure_list(append_conflicts)
remove_conflicts = ensure_list(remove_conflicts)
def check_target_directory(target_directory, source_control):
if os.path.isdir(target_directory) and os.listdir(target_directory):
pe = PackageEntry().load_control_from_wapt(target_directory)
if pe.package != source_control.package or pe > source_control:
raise Exception('Target directory "%s" is not empty and contains either another package or a newer version, aborting.' % target_directory)
# duplicate a development directory tree
if os.path.isdir(packagename):
source_control = PackageEntry().load_control_from_wapt(packagename)
if not newname:
newname = source_control.package
if target_directory == '':
target_directory = self.get_default_development_dir(newname, section=source_control.section)
if target_directory is None:
target_directory = tempfile.mkdtemp('wapt')
# check if we will not overwrite newer package or different package
check_target_directory(target_directory, source_control)
if packagename != target_directory:
shutil.copytree(packagename, target_directory)
# duplicate a wapt file
elif os.path.isfile(packagename):
source_filename = packagename
source_control = PackageEntry().load_control_from_wapt(source_filename)
if not newname:
newname = source_control.package
if target_directory == '':
target_directory = self.get_default_development_dir(newname, section=source_control.section)
if target_directory is None:
target_directory = tempfile.mkdtemp('wapt')
# check if we will not overwrite newer package or different package
check_target_directory(target_directory, source_control)
source_control.unzip_package(target_dir=target_directory, cabundle=cabundle)
else:
source_package = self.is_available(packagename)
if not source_package:
raise Exception('Package %s is not available in current repositories.' % (packagename,))
# duplicate package from a repository
filenames = self.download_packages([packagename], usecache=usecache, printhook=printhook)
package_paths = filenames['downloaded'] or filenames['skipped']
if not package_paths:
raise Exception('Unable to download package %s' % (packagename,))
source_filename = package_paths[0]
source_control = PackageEntry().load_control_from_wapt(source_filename)
if not newname:
newname = source_control.package
if target_directory == '':
target_directory = self.get_default_development_dir(newname, section=source_control.section)
if target_directory is None:
target_directory = tempfile.mkdtemp('wapt')
# check if we will not overwrite newer package or different package
check_target_directory(target_directory, source_control)
source_control.unzip_package(target_dir=target_directory, cabundle=cabundle)
# duplicate package informations
dest_control = PackageEntry()
for a in source_control.required_attributes + source_control.optional_attributes:
dest_control[a] = source_control[a]
if newmaturity is not None:
dest_control.maturity = newmaturity
else:
dest_control.maturity = self.default_maturity
# add / remove dependencies from copy
prev_depends = ensure_list(dest_control.depends)
for d in append_depends:
if not d in prev_depends:
prev_depends.append(d)
for d in remove_depends:
if d in prev_depends:
prev_depends.remove(d)
dest_control.depends = ','.join(prev_depends)
# add / remove conflicts from copy
prev_conflicts = ensure_list(dest_control.conflicts)
for d in append_conflicts:
if not d in prev_conflicts:
prev_conflicts.append(d)
for d in remove_conflicts:
if d in prev_conflicts:
prev_conflicts.remove(d)
dest_control.conflicts = ','.join(prev_conflicts)
# change package name
dest_control.package = newname
if newversion:
dest_control.version = newversion
# Check existing versions of newname and increment it
if auto_inc_version:
older_packages = self.is_available(newname)
if older_packages and dest_control <= older_packages[-1]:
dest_control.version = older_packages[-1].version
dest_control.inc_build()
dest_control.filename = dest_control.make_package_filename()
dest_control.save_control_to_wapt(target_directory)
if dest_control.section != 'host':
self.add_pyscripter_project(target_directory)
self.add_vscode_project(target_directory)
dest_control.invalidate_signature()
return dest_control
[docs] def set_package_attribute(self, package, key, value, public=False):
"""Store in local db a key/value pair for later use"""
self.write_param(package+'.'+key, value, public=public)
[docs] def get_package_attribute(self, package, key, default_value=None, public=False):
"""Store in local db a key/value pair for later use"""
return self.read_param(package+'.'+key, default_value, public=public)
[docs] def read_param(self, name, default=None, ptype=None, public=None):
"""read a param value from local db
if public==True, try first in public db then in private
this handle the case where private db is not accessible by non admin users.
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> wapt.read_param('db_version')
u'20140410'
"""
if public is None:
public = name in self.public_params
if public and self.waptpublicdb.has_param(name):
return self.waptpublicdb.get_param(name, default, ptype)
return self.waptdb.get_param(name, default, ptype)
[docs] def write_param(self, name, value, public=None):
"""Store in local db a key/value pair for later use"""
if public is None:
public = name in self.public_params
if public:
with self.waptdb, self.waptpublicdb:
self.waptdb.set_param(name, value)
self.waptpublicdb.set_param(name, value)
else:
with self.waptdb:
self.waptdb.set_param(name, value)
[docs] def delete_param(self, name, public=True):
"""Remove a key from local db"""
if public:
with self.waptdb,self.waptpublicdb:
self.waptdb.delete_param(name)
self.waptpublicdb.delete_param(name)
else:
with self.waptdb,self.waptpublicdb:
self.waptdb.delete_param(name)
[docs] def dependencies(self, packagename, expand=False):
"""Return all dependecies of a given package
>>> w = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> dep = w.dependencies('tis-waptdev')
>>> isinstance(dep,list) and isinstance(dep[0],PackageEntry)
True
"""
packages = self.is_available(packagename)
result = []
errors = []
if packages:
depends = ensure_list(packages[-1].depends)
for dep in depends:
subpackages = self.is_available(dep)
if subpackages:
if expand:
result.extend(self.dependencies(dep))
if not subpackages[-1] in result:
result.append(subpackages[-1])
else:
errors.append(dep)
return result
[docs] def get_package_entries(self, packages_names):
r"""Return most up to date packages entries for packages_names
packages_names is either a list or a string.
'missing' key lists the package requirements which are not available in the
package index.
Args;
packages_names (list or str): list of package requirements
Returns:
dict : {'packages':[PackageEntries,],'missing':[str,]}
>>> wapt = Wapt(config_filename='c:/wapt/wapt-get.ini')
>>> res = wapt.get_package_entries(['tis-firefox','tis-putty'])
>>> isinstance(res['missing'],list) and isinstance(res['packages'][0],PackageEntry)
True
"""
result = {'packages': [], 'missing': []}
if isinstance(packages_names, str) or isinstance(packages_names, str):
packages_names = [p.strip() for p in packages_names.split(",")]
for package_name in packages_names:
matches = self.waptdb.packages_matching(package_name)
if matches:
result['packages'].append(matches[-1])
else:
result['missing'].append(package_name)
return result
[docs] def add_upgrade_shutdown_policy(self):
"""Add a local shitdown policy to upgrade system"""
waptexit_path = setuphelpers.makepath(self.wapt_base_dir, 'waptexit.exe')
if not os.path.isfile(waptexit_path):
raise Exception('Can not find %s' % waptexit_path)
setuphelpers.shutdown_scripts_ui_visible(state=True)
return setuphelpers.add_shutdown_script(waptexit_path, '')
[docs] def remove_upgrade_shutdown_policy(self):
"""Add a local shitdown policy to upgrade system"""
waptexit_path = setuphelpers.makepath(self.wapt_base_dir, 'waptexit.exe')
if not os.path.isfile(waptexit_path):
raise Exception('Can not find %s' % waptexit_path)
return setuphelpers.remove_shutdown_script(waptexit_path, '')
[docs] def show_progress(self, show_box=False, msg='Loading...', progress=None, progress_max=None):
"""Global hook to report progress feedback to the user
Args:
show_box (bool): indicate to display or hide the notification
msg (str): A status message to display. If None, nothing is changed.
progress (float): Completion
progress_max (float): Target of completion.
"""
if self.progress_hook:
return self.progress_hook(show_box, msg, progress, progress_max) # pylint: disable=not-callable
else:
print(('%s : %s / %s' % (msg, progress, progress_max)))
return False
[docs] def get_secured_token_generator(self,token_secret_key):
return URLSafeTimedSerializer(token_secret_key)
[docs] def is_authorized_package_action(self, action, package_cond, user_groups=[], rules=None):
package_request = PackageRequest(request=package_cond)
if package_request.package_uuid:
pe = self.waptdb.packages_matching(package_request)
if pe:
package = pe[0].package
else:
return False
else:
package = package_request.package
if not package:
return False
if action in ('install', 'upgrade'):
if package in self.installed_package_names():
return True
upgrades_and_pending = [PackageRequest(pr).package for pr in self.get_last_update_status().get('upgrades', [])]
if package in upgrades_and_pending :
return True
if not user_groups:
return False
if 'waptselfservice' in user_groups:
return True
if rules is None and self.is_enterprise():
rules = self_service_rules(self)
for group in user_groups:
if package in rules.get(group, []):
return True
return False
[docs] def available_categories(self):
return list(set([k.get('keywords').capitalize().split(',')[0] for k in self.waptdb.query('select distinct keywords from wapt_package where keywords is not null')]))
[docs] def self_service_auth(self, login, password, host_uuid, groups):
""" Sends login and password to server, who then checks if it's a valid user
Returns:
list: groups the user belongs to.
"""
result = None
if not self.waptserver_available():
raise Exception("Waptserver is not available ; cannot send credentials to server")
try:
data = {"user": login,
"password": password,
"uuid": host_uuid,
"groups": list(groups)
}
signature = self.sign_host_content(str.encode(jsondump(data)))
result = self.waptserver.post('login_self_service',
data=jsondump(data),
signature=signature,
signer=self.get_host_certificate().cn
)
if result and result['success']:
logger.info('User successfully authenticated %s updated properly' % self.waptserver.server_url)
return result
else:
raise Exception('Error authenticating on server %s: %s' % (self.waptserver.server_url, result and result['msg'] or 'No message'))
except requests.HTTPError as e:
logger.debug('Unable to authenticate on server : %s' % traceback.format_exc())
logger.warning('Unable to authenticate on server : %s' % e)
raise e
### audit / metrics related dara
def _audit_data_to_db(self,value):
if value is None:
return None
if isinstance(value, datetime.datetime):
value = datetime2isodate(value)
value = jsondump(value)
return value
def _audit_data_from_db(self, value, ptype=None):
if value is None:
return None
else:
# stored always in json format
try:
value = ujson.loads(value)
except ValueError:
# tolerant for old data
pass
# specific output conversion
if ptype == 'datetime':
return isodate2datetime(value)
return value
[docs] def audit_data_expired(self, section, key, value):
"""Check if the latest value associated with section/key is expired
Returns:
bool: True is data exists and has expires or if data does not exists.
"""
q = self.waptdb.execute('select expiration_date from wapt_audit_data where value_section=? and value_key=? order by value_date desc limit 1', (section,key,)).fetchone()
if q:
(expiration_date, ) = q
return not expiration_date or expiration_date < datetime2isodate(datetime.datetime.utcnow())
else:
return True
[docs] def write_audit_data_if_changed(self, section, key, value, ptype=None, value_date=None, expiration_date=None, max_count=2, keep_days=None):
"""Write data only if different from last one
Returns:
previous value
"""
previous = self.read_audit_data(section, key, include_expired_data=False)
if previous != value:
self.write_audit_data(section=section, key=key, value=value, value_date=value_date, expiration_date=expiration_date, max_count=max_count, keep_days=keep_days)
return previous
[docs] def write_audit_data(self, section, key, value, ptype=None, value_date=None, expiration_date=None, max_count=2, keep_days=None):
"""Stores in database a metrics, removes expired ones
Args:
section (str)
key (str)
value (any)
value_date (str or datetime): value date (utc). By default datetime.datetime.utcnow()
expiration_date (str) : expiration date of the new value
max_count (int) : keep at most max_count value. remove oldest one.
keep_days (int) : set the expiration date to now + keep_days days. override expiration_date arg if not None
Returns:
None
"""
with self.waptdb:
value = self._audit_data_to_db(value)
# if value_date is not provided, use current timestamp
if value_date is None:
value_date = datetime2isodate(datetime.datetime.utcnow())
if isinstance(value_date, datetime.datetime):
value_date = datetime2isodate(value_date)
if keep_days:
expiration_date = datetime2isodate(datetime.datetime.utcnow() + datetime.timedelta(days=keep_days))
self.waptdb.execute('insert or replace into wapt_audit_data(value_date,value_section,value_key,value,expiration_date) values (?,?,?,?,?)',
(value_date,section,key,value,expiration_date))
# removes expired values
# current count of metric
if max_count is not None:
cur = self.waptdb.execute("""select min(value_date) from (select value_date from wapt_audit_data where value_section=? and value_key=? order by value_date desc limit ?)""",(section,key,max_count))
if cur:
(min_value_date,) = cur.fetchone()
# delete oldest in order to keep at least max_count
self.waptdb.execute("""delete from wapt_audit_data where value_date<? and value_section=? and value_key=?""",
(min_value_date,section,key))
self.waptdb.execute("""delete from wapt_audit_data where value_date<? and value_section=? and value_key=? and (expiration_date is null or expiration_date<?)""",
(value_date,section,key,datetime2isodate(datetime.datetime.utcnow())))
[docs] def read_audit_data(self, section, key, default=None, ptype=None, include_expired_data=True):
"""Retrieve the latest value associated with section/key from database"""
if include_expired_data:
expiration_date = '0000-00-00'
else:
expiration_date = datetime2isodate(datetime.datetime.utcnow())
q = self.waptdb.execute('select value from wapt_audit_data where value_section=? and value_key=? and (expiration_date is null or expiration_date >= ?) order by value_date desc limit 1', (section,key,expiration_date)).fetchone()
if q:
(value, ) = q
value = self._audit_data_from_db(value,ptype)
if value is None:
value = default
return value
else:
return default
[docs] def read_audit_data_set(self, section, key, as_dict=False, raw_data=False, descending=True):
"""Retrieve all the values associated with section/key from database"""
if descending:
desc='desc'
else:
desc=''
for (value, value_date, expiration_date) in self.waptdb.execute("""\
select value,value_date,expiration_date from wapt_audit_data
where value_section=? and value_key=?
order by value_date %s
""" % desc, (section,key)).fetchall():
if raw_data:
if as_dict:
yield dict(value=value, value_date=value_date, expiration_date=expiration_date)
else:
yield (value, value_date, expiration_date)
else:
if as_dict:
yield dict(value=self._audit_data_from_db(value), value_date=value_date, expiration_date=expiration_date)
else:
yield (self._audit_data_from_db(value), value_date, expiration_date)
[docs] def delete_audit_data(self, section, key):
with self:
row = self.waptdb.execute('select value from wapt_audit_data where value_section=? and value_key like ? limit 1', (section,key,)).fetchone()
if row:
self.waptdb.execute('delete from wapt_audit_data where value_section=? and value_key like ?', (section,key,))
[docs] def read_audit_data_since(self, last_query_date=None,raw_data=False):
"""Retrieve all the values associated with section/key from database"""
if last_query_date is None:
last_query_date=''
yield ('id','value_section', 'value_key', 'value_date', 'value', 'expiration_date')
for (id,value_section, value_key, value_date, value, expiration_date) in self.waptdb.execute("""\
select id,value_section, value_key, value_date, value, expiration_date from wapt_audit_data
where value_date > ?
order by value_date
""", (last_query_date,)).fetchall():
if raw_data:
yield (
id,
value_section,
value_key,
value_date,
value,
expiration_date)
else:
yield (
id,
value_section,
value_key,
value_date,
self._audit_data_from_db(value),
expiration_date)
[docs] def get_next_audit_datetime(self):
"""Return next datetime for next audit loop
= minimum(next_audit_date)
"""
with self.waptdb:
query = self.waptdb.query("select min(next_audit_on) as next_audit from wapt_localstatus l where l.install_status <> 'ERROR' and (next_audit_on is not null and next_audit_on<>'') and (next_audit_on > last_audit_on)")
if query:
isots = query[0]['next_audit']
if isots:
d = isodate2datetime(isots)
d_stripseconds = datetime.datetime(d.year,d.month,d.day,d.hour,d.minute) # strip the seconds
return d_stripseconds
else:
return None
else:
return None
[docs] def call_python_code(self, python_filename, func_name, package_entry=None, force = None, params=None, working_dir = None, import_modules = []):
"""Calls a function in python_filename.
Set basedir, control, and run context within the function context.
Args:
python_filename : python filename mith module to load.
func_name (str): name of function to call in setuppy
package_entry (PackageEntry): if not None, use it to set environment
Returns:
output of hook.
"""
if not os.path.isfile(python_filename):
raise Exception('Python file not found: %s, aborting.' % ensure_unicode(python_filename))
if working_dir is None:
working_dir = os.path.abspath(os.path.dirname(python_filename))
# we record old sys.path as we will include current setup.py
oldpath = sys.path
try:
previous_cwd = os.getcwd()
os.chdir(working_dir)
# import the setup module from package file
logger.info(" sourcing py file %s " % ensure_unicode(python_filename))
# import code as file to allow debugging.
setup = import_setup(python_filename)
hook_func = getattr(setup, func_name, None)
if hook_func is None:
raise EWaptMissingPackageHook('No %s function found in %s module' % (func_name, python_filename))
try:
# import all names from modules.
for module_name in import_modules:
try:
module_info = imp.find_module(module_name)
code = module_info[0].read()
exec(code, setup.__dict__)
except ImportError as e:
logger.critical('Unable to import implicit module %s : %s' % (module_name,e))
if package_entry:
package_entry._set_hook_module_environment(setup, wapt_context=self, params=params, force=force, user=self.user)
logger.info(" executing %s" % (func_name, ))
with _disable_file_system_redirection():
hookdata = hook_func()
return hookdata
except Exception as e:
logger.critical('Fatal error in %s %s:\n%s' % (func_name, ensure_unicode(e), ensure_unicode(traceback.format_exc())))
raise e
finally:
os.chdir(previous_cwd)
gc.collect()
if 'setup' in dir() and setup is not None:
setup_name = setup.__name__[:]
logger.debug('Removing module: %s, refcnt: %s' % (setup_name, sys.getrefcount(setup)))
del setup
if setup_name in sys.modules:
del sys.modules[setup_name]
sys.path = oldpath
[docs] def get_json_config_filename(self,config_name):
"""Returns the filename for a json config named <config_name>
"""
return setuphelpers.makepath(self.configs_dir, config_name + ".json")
[docs] def install_json_config(self, conf, config_name=None, priority=None):
"""Add a dynamic configuration from dict conf with name config_name
and priority
"""
try:
if not type(conf) == dict:
raise EWaptException('JSON Config object must be a dict not a %s' % (type(conf)))
if config_name is None:
config_name = conf.get('name')
if not config_name:
raise EWaptException('Unable to install json config, no config_name')
# awful hack, as repo-sync is not a valid identifier, we store 'reposync' key in json
# but keep [repo-sync] section name in ini file for compat.
if 'reposync' in conf:
conf['repo-sync'] = conf['reposync']
del conf['reposync']
if not type(conf) is dict:
raise Exception('Json configuration is not a dict')
if priority:
conf['priority'] = int(priority)
# first remove previous files
self.remove_json_config(config_name)
# extract packages signer certificates
crts = conf.get('certificates', {})
for key in crts:
crt_dest = setuphelpers.makepath(self.public_certs_dir, "%s-%s.crt" % (config_name, key))
with open(crt_dest, "w") as f:
f.write(crts[key])
for section in conf:
# these are managed specifically
if section in ('signature', 'filename', 'priority', 'name', 'certificates', 'server_certificates'):
continue
if not isinstance(conf[section],dict):
continue
# change verify_cert fileid into a filename
if conf[section].get('verify_cert', None):
if conf[section]['verify_cert'].lower().strip() in ['0','1','true','false']:
conf[section]['verify_cert'] = conf[section]['verify_cert']
else:
# extract TLS server certificates
cert_server = setuphelpers.makepath(self.public_certs_dir,'server',config_name + '-' + conf[section]['verify_cert'] + '.crt')
with open(cert_server, "w") as f:
f.write(conf['server_certificates'][conf[section]['verify_cert']])
conf[section]['verify_cert'] = cert_server
# Copy the json config as a josn file to <wapt>/conf.d
config_filename = self.get_json_config_filename(config_name)
with open(config_filename, "w") as config_file:
config_file.write(json.dumps(conf))
return (os.path.isfile(config_filename), config_filename)
except Exception as e:
raise Exception("Invalid json configuration %s:\n%s" % (config_name, str(e)))
[docs] def install_json_configb64(self,json_config_b64, config_name=None, priority=None):
"""Install a json config encoded as a base64 string"""
try:
json_config = json.loads(base64.b64decode(json_config_b64.encode('utf-8')).decode('utf-8'))
return self.install_json_config(json_config, config_name, priority)
except:
raise Exception("Invalid base64 json configuration : \n%s" % json_config_b64)
[docs] def install_json_config_from_url(self,url=None, config_hash=None, config_name='default_config', priority=None):
"""Load a json config from the remote wapt server (default location is /var/www/wapt/conf.d/<config_name>-<config_hash>.json)
given its name and hash.
"""
if url is None and self.waptserver:
url = "wapt/conf.d/%s_%s.json" % (config_name, config_hash)
json_config = self.waptserver.get(url,decode_json=False)
elif url is None and self.repositories:
url = "%s/conf.d/%s_%s.json" % (self.repositories[0].repo_url, config_name, config_hash)
with self.repositories[0].get_requests_session as session:
json_config = session.get(url).content
return self.install_json_config(json_config, config_name, priority)
else:
json_config = wgets(url,verify_cert = not config_hash)
if sha256_for_data(json_config) != config_hash:
raise EWaptException('Bad sha256 checksum for config from url %s' % url)
res = self.install_json_config(json.loads(json_config), config_name, priority=priority)
return res
[docs] def install_json_config_file(self,json_config_file, config_name=None, priority=None):
"""Install a json config stored in a file"""
if not os.path.isfile(json_config_file):
raise Exception("%s configuration file not found" % json_config_file)
with open(json_config_file, "r") as file:
json_config = json.load(file)
return self.install_json_config(json_config, config_name, priority)
[docs] def get_json_config_certificates_filenames(self,config_name):
# get the list of filenames of signers certificates provided by this configuration
config_filename = self.get_json_config_filename(config_name)
result = []
if os.path.isfile(config_filename):
with open(config_filename, "r") as config_file:
json_config = json.load(config_file)
crts = json_config.get('certificates', [])
for key in crts:
crt_filename = setuphelpers.makepath(self.public_certs_dir, "%s-%s.crt" % (config_name, key))
result.append(crt_filename)
return result
[docs] def remove_json_config(self,config_name):
"""Remove a json config given its name"""
previous_json = None
filename = self.get_json_config_filename(config_name)
if os.path.isfile(filename):
# load the config as a dict from json file.
with open(filename, "r") as file:
json_config = json.load(file)
previous_json = file.read()
os.remove(filename)
# remove the package signers certificates
crts = json_config.get('certificates', [])
for key in crts:
crt_filename = setuphelpers.makepath(self.public_certs_dir, "%s-%s.crt" % (config_name, key))
if os.path.isfile(crt_filename):
os.unlink(crt_filename)
# loop over the sections to remove the TLS certificates
for section in json_config:
# these are managed specifically
if section in ('signature', 'filename', 'priority', 'name', 'certificates', 'server_certificates'):
continue
if not isinstance(json_config[section],dict):
continue
if json_config[section].get('verify_cert'):
verify_cert = json_config[section]['verify_cert']
if not verify_cert.lower().strip() in ['0','1','true','false']:
verify_cert_filename = setuphelpers.makepath(self.public_certs_dir,'server','%s-%s.crt' % (config_name,verify_cert))
if os.path.isfile(verify_cert_filename):
os.unlink(verify_cert_filename)
return previous_json
[docs] def installed_package_names(self, include_errors=False):
"""Private list of installed packages
"""
sql = ["select l.package from wapt_localstatus l"]
if not include_errors:
sql.append('where l.install_status in ("OK","UNKNOWN")')
return [p['package'] for p in self.waptdb.query('\n'.join(sql))]
[docs] def public_installed_package_names(self):
"""Public list of installed packages
"""
sql = "select distinct l.package from wapt_publicstatus l where install_date is not null and uninstall_date is null order by install_date desc"
return [p['package'] for p in self.waptpublicdb.query(sql)]
[docs] def check_peercache_status(self):
if not waptlicences:
return
try:
started = False
status = waptlicences.peercache_status()
if status is None:
if self.peercache_enable:
started = waptlicences.peercache_init(self.host_uuid, self.config_filename)
else:
if status['running']:
waptlicences.peercache_stop()
if self.peercache_enable:
started = waptlicences.peercache_start()
logger.info('check_peercache_status: Peercache started ?: %s' % started)
return started
except Exception as e:
logger.warning('Unable to update peercache state: %s' % e)
[docs]def wapt_sources_edit(wapt_sources_dir: str, editor_for_packages: str = None) -> str:
r"""Utility to open PyScripter or the configured editor with package sources if it is installed
else open the wapt package development directory in System Shell Explorer.
Args
wapt_sources_dir (str): directory path of the wapt package sources
Returns:
str: sources path
"""
wapt_sources_dir = ensure_unicode(wapt_sources_dir)
if waptlicences:
return waptlicences.edit_wapt_sources(wapt_sources_dir, editor_for_packages=editor_for_packages or '')
params = {
"wapt_base_dir": os.path.dirname(__file__),
"wapt_sources_dir": wapt_sources_dir,
"setup_filename": os.path.join(wapt_sources_dir, "setup.py"),
"control_filename": os.path.join(wapt_sources_dir, "WAPT", "control"),
"changelog_filename": os.path.join(wapt_sources_dir, "WAPT", "changelog.txt"),
"update_package_filename": os.path.join(wapt_sources_dir, "update_package.py"),
}
params_vscod_list = [params["wapt_sources_dir"], params["setup_filename"], params["control_filename"], params["update_package_filename"], params["changelog_filename"]]
# in edit_for_packages you can specify {key_params} to replace for launch the editor
env = os.environ
env.update(dict(PYTHONPATH=params["wapt_base_dir"], VIRTUAL_ENV=params["wapt_base_dir"]))
if os.name == "nt":
if not (editor_for_packages) or editor_for_packages == "pyscripter":
pyscripter_filename = os.path.join(setuphelpers.programfiles32, "PyScripter", "PyScripter.exe")
if sys.platform == "win32" and os.path.isfile(pyscripter_filename):
params["psproj_filename"] = os.path.join(wapt_sources_dir, "WAPT", "wapt.psproj")
setuphelpers.run_as_administrator(
pyscripter_filename,
'--PYTHONDLLPATH="{wapt_base_dir}" --python39 -N --project="{psproj_filename}" "{setup_filename}" "{control_filename}" "{update_package_filename}" "{changelog_filename}"'.format(
**params
),
)
elif shutil.which("code"):
command = [shutil.which("code"), *params_vscod_list]
run(command)
elif shutil.which("codium"):
command = [shutil.which("codium"), *params_vscod_list]
run(command)
else:
os.startfile(params["wapt_sources_dir"])
else:
try:
exe_file = ""
if editor_for_packages.strip("vs") in ["code", "codium", "vscode", "vscodium"] and os.path.isfile(
shutil.which(editor_for_packages.strip("vs"))
):
exe_file = shutil.which(editor_for_packages.strip("vs"))
elif editor_for_packages.find(".exe") != -1 and os.path.isfile(editor_for_packages[: editor_for_packages.find(".exe") + 4]):
exe_position = editor_for_packages.find(".exe")
exe_file = editor_for_packages[: exe_position + 4]
if exe_file:
command = [exe_file, *params_vscod_list]
run(command)
else:
os.startfile(params["wapt_sources_dir"])
except:
os.startfile(params["wapt_sources_dir"])
else:
command = []
list_supported_editor = ["codium", "vscodium", "vscode", "code", "nano", "vim", "vi"]
if (editor_for_packages is not None) and (editor_for_packages not in list_supported_editor):
space_sep = editor_for_packages.find(" ")
params_string = editor_for_packages[space_sep + 1 :].format(**params)
command = [editor_for_packages, params_string]
elif shutil.which("codium") and ((editor_for_packages is None) or (editor_for_packages in ["codium", "vscodium"])):
command = ["codium", *params_vscod_list]
elif shutil.which("code") and ((editor_for_packages is None) or (editor_for_packages in ["code", "vscode"])):
command = ["code", *params_vscod_list]
elif shutil.which("nano") and ((editor_for_packages is None) or (editor_for_packages == "nano")):
command = ["nano", params["setup_filename"]]
elif shutil.which("vim") and ((editor_for_packages is None) or (editor_for_packages == "vim")):
command = ["vim", params["setup_filename"]]
elif shutil.which("vi") and ((editor_for_packages is None) or (editor_for_packages == "vi")):
command = ["vi", params["setup_filename"]]
if command:
subprocess.call(command)
return wapt_sources_dir
[docs]def sid_from_rid(domain_controller, rid):
"""Return SID structure based on supplied domain controller's domain and supplied rid
rid can be for example DOMAIN_GROUP_RID_ADMINS, DOMAIN_GROUP_RID_USERS
"""
umi2 = win32net.NetUserModalsGet(domain_controller, 2)
domain_sid = umi2['domain_id']
sub_authority_count = domain_sid.GetSubAuthorityCount()
# create and init new sid with acct domain Sid + acct rid
sid = pywintypes.SID()
sid.Initialize(domain_sid.GetSidIdentifierAuthority(),
sub_authority_count+1)
# copy existing subauthorities from account domain Sid into
# new Sid
for i in range(sub_authority_count):
sid.SetSubAuthority(i, domain_sid.GetSubAuthority(i))
# append Rid to new Sid
sid.SetSubAuthority(sub_authority_count, rid)
return sid
[docs]def lookup_name_from_rid(domain_controller, rid):
""" return username or group name from RID (with localization if applicable)
from https://mail.python.org/pipermail/python-win32/2006-May/004655.html
domain_controller : should be a DC
rid : integer number (512 for domain admins, 513 for domain users, etc.)
>>> lookup_name_from_rid('srvads', DOMAIN_GROUP_RID_ADMINS)
u'Domain Admins'
"""
sid = sid_from_rid(domain_controller, rid)
name, domain, typ = win32security.LookupAccountSid(domain_controller, sid)
return name
[docs]def get_domain_admins_group_name():
r"""Return localized version of domain admin group (ie "domain admins" or
"administrateurs du domaine" with RID -512)
>>> get_domain_admins_group_name()
u'Domain Admins'
"""
try:
target_computer = win32net.NetGetAnyDCName()
name = lookup_name_from_rid(target_computer, DOMAIN_GROUP_RID_ADMINS)
return name
except Exception as e:
logger.debug('Error getting Domain Admins group name : %s' % e)
return 'Domain Admins'
[docs]def get_local_admins_group_name():
sid = win32security.GetBinarySid('S-1-5-32-544')
name, domain, typ = win32security.LookupAccountSid(setuphelpers.wincomputername(), sid)
return name
[docs]def check_is_member_of(huser, group_name):
"""Check if a user is a member of a group
Args:
huser (handle) : pywin32
group_name (str) : group
>>> from win32security import LogonUser
>>> hUser = win32security.LogonUser ('technique','tranquilit','xxxxxxx',win32security.LOGON32_LOGON_NETWORK,win32security.LOGON32_PROVIDER_DEFAULT)
>>> check_is_member_of(hUser,'domain admins')
False
"""
try:
sid, system, type = win32security.LookupAccountName(None, group_name)
except:
logger.debug('"%s" is not a valid group name' % group_name)
return False
return win32security.CheckTokenMembership(huser, sid)
[docs]def check_user_membership(user_name, password, domain_name, group_name):
"""Check if a user is a member of a group
Args:
user_name (str): user
password (str):
domain_name (str) : If empty, check local then domain
group_name (str): group
>>> from win32security import LogonUser
>>> hUser = win32security.LogonUser ('technique','tranquilit','xxxxxxx',win32security.LOGON32_LOGON_NETWORK,win32security.LOGON32_PROVIDER_DEFAULT)
>>> check_is_member_of(hUser,'domain admins')
False
"""
try:
sid, system, type = win32security.LookupAccountName(None, group_name)
except pywintypes.error as e:
if e.args[0] == 1332:
logger.warning('"%s" is not a valid group name' % group_name)
return False
else:
raise
huser = win32security.LogonUser(user_name, domain_name, password, win32security.LOGON32_LOGON_NETWORK, win32security.LOGON32_PROVIDER_DEFAULT)
return win32security.CheckTokenMembership(huser, sid)
[docs]def self_service_rules(wapt: Wapt):
"""Returns dict of allowed packages for users and groups
"""
cur = wapt.waptdb.execute("""select package, package_uuid from wapt_localstatus s where s.section='selfservice' and s.package_uuid is not null""")
result = {}
for package, package_uuid in cur.fetchall():
if is_unsafe_filename(package_uuid):
continue
rules_fn = setuphelpers.makepath(wapt.persistent_root_dir, package_uuid, 'selfservice.json')
if os.path.isfile(rules_fn):
with open(rules_fn, 'r') as f:
rules = json.load(f)
for group, packages in rules.items():
if not group.lower() in result:
result[group.lower()] = packages
else:
group_packages = result[group.lower()]
for package in packages:
if not package in group_packages:
group_packages.append(package)
return result
[docs]def authorized_packages_for_token(wapt, token, secret_key, package_requests=None):
token_gen = wapt.get_secured_token_generator(secret_key)
max_age = wapt.token_lifetime
user_pac = token_gen.loads(token, max_age=max_age)
user_groups = user_pac.get('groups', [])
rules = self_service_rules(wapt)
result = []
if package_requests is None:
for group in user_groups:
for package in rules.get(group, []):
if not package in result:
result.append(package)
else:
for package_request in package_requests:
if isinstance(package_request, PackageRequest):
pr = package_request
elif isinstance(package_request, str):
pr = PackageRequest(pr)
elif isinstance(package_request, PackageEntry):
pr = package_request.as_package_request()
else:
continue
for group in user_groups:
if pr.package in rules.get(group, []):
result.append(pr)
break
return result
# for backward compatibility
Version = setuphelpers.Version # obsolete
if __name__ == '__main__':
sys.exit(0)