# -*- coding: utf-8 -*- # Copyright (c) 2016 Red Hat, Inc. # # Permission is hereby granted, free of charge, to any person obtaining a copy # of this software and associated documentation files (the "Software"), to deal # in the Software without restriction, including without limitation the rights # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell # copies of the Software, and to permit persons to whom the Software is # furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice shall be included in all # copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE # SOFTWARE. # # Written by Matt Prahl except for the test functions from __future__ import print_function from flask_script import Manager from functools import wraps import flask_migrate import logging import os import getpass from werkzeug.datastructures import FileStorage from module_build_service import app, conf, db, create_app from module_build_service import models from module_build_service.utils import ( submit_module_build_from_yaml, load_local_builds, load_mmd, import_mmd ) from module_build_service.errors import StreamAmbigous import module_build_service.messaging import module_build_service.scheduler.consumer manager = Manager(create_app) help_args = ('-?', '--help') manager.help_args = help_args migrate = flask_migrate.Migrate(app, db) manager.add_command('db', flask_migrate.MigrateCommand) manager.add_option('-d', '--debug', dest='debug', action='store_true') manager.add_option('-v', '--verbose', dest='verbose', action='store_true') manager.add_option('-q', '--quiet', dest='quiet', action='store_true') def console_script_help(f): @wraps(f) def wrapped(*args, **kwargs): import sys if any([arg in help_args for arg in sys.argv[1:]]): command = os.path.basename(sys.argv[0]) print("""{0} Usage: {0} [{1}] See also: mbs-manager(1)""".format(command, '|'.join(help_args))) sys.exit(2) r = f(*args, **kwargs) return r return wrapped @console_script_help @manager.command def upgradedb(): """ Upgrades the database schema to the latest revision """ app.config["SERVER_NAME"] = 'localhost' # TODO: configurable? migrations_dir = os.path.join(os.path.abspath(os.path.dirname(__file__)), 'migrations') with app.app_context(): flask_migrate.upgrade(directory=migrations_dir) @manager.command def cleardb(): """ Clears the database """ models.ModuleBuild.query.delete() models.ComponentBuild.query.delete() @console_script_help @manager.command def import_module(mmd_file): """ Imports the module from mmd_file """ mmd = load_mmd(mmd_file, is_file=True) import_mmd(db.session, mmd) @manager.option('--stream', action='store', dest="stream") @manager.option('--file', action='store', dest="yaml_file") @manager.option('--skiptests', action='store_true', dest="skiptests") @manager.option('-l', '--add-local-build', action='append', default=None, dest='local_build_nsvs') @manager.option('-s', '--set-stream', action='append', default=[], dest='default_streams') def build_module_locally(local_build_nsvs=None, yaml_file=None, stream=None, skiptests=False, default_streams=None): """ Performs local module build using Mock """ if 'SERVER_NAME' not in app.config or not app.config['SERVER_NAME']: app.config["SERVER_NAME"] = 'localhost' if app.config['RESOLVER'] == 'db': raise ValueError("Please set RESOLVER to 'mbs' in your " "configuration for local builds.") with app.app_context(): conf.set_item("system", "mock") # Use our own local SQLite3 database. confdir = os.path.abspath(os.getcwd()) dbdir = os.path.abspath(os.path.join(confdir, '..')) if confdir.endswith('conf') \ else confdir dbpath = '/{0}'.format(os.path.join(dbdir, '.mbs_local_build.db')) dburi = 'sqlite://' + dbpath app.config["SQLALCHEMY_DATABASE_URI"] = dburi conf.set_item("sqlalchemy_database_uri", dburi) if os.path.exists(dbpath): os.remove(dbpath) db.create_all() load_local_builds(local_build_nsvs) optional_params = {} optional_params["local_build"] = True optional_params["default_streams"] = {} for ns in default_streams: name, stream = ns.split(":") optional_params["default_streams"][name] = stream username = getpass.getuser() if not yaml_file or not yaml_file.endswith(".yaml"): raise IOError("Provided modulemd file is not a yaml file.") yaml_file_path = os.path.abspath(yaml_file) with open(yaml_file_path) as fd: filename = os.path.basename(yaml_file) handle = FileStorage(fd) handle.filename = filename try: submit_module_build_from_yaml( username, handle, str(stream), skiptests, optional_params) except StreamAmbigous as e: logging.error(str(e)) logging.error( "Use '-s module_name:module_stream' to choose the stream") return stop = module_build_service.scheduler.make_simple_stop_condition(db.session) # Run the consumer until stop_condition returns True module_build_service.scheduler.main([], stop) @console_script_help @manager.command def run(host=None, port=None, debug=None): """ Runs the Flask app, locally. """ host = host or conf.host port = port or conf.port debug = debug or conf.debug logging.info('Starting Module Build Service frontend') app.run( host=host, port=port, debug=debug ) def manager_wrapper(): manager.run() if __name__ == "__main__": manager_wrapper()