aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'buildbot_gentoo_ci/db')
-rw-r--r--buildbot_gentoo_ci/db/__init__.py0
-rw-r--r--buildbot_gentoo_ci/db/connector.py98
-rw-r--r--buildbot_gentoo_ci/db/migrate/README4
-rw-r--r--buildbot_gentoo_ci/db/migrate/migrate.cfg20
-rw-r--r--buildbot_gentoo_ci/db/migrate/versions/__init__.py0
-rw-r--r--buildbot_gentoo_ci/db/model.py352
6 files changed, 474 insertions, 0 deletions
diff --git a/buildbot_gentoo_ci/db/__init__.py b/buildbot_gentoo_ci/db/__init__.py
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/buildbot_gentoo_ci/db/__init__.py
diff --git a/buildbot_gentoo_ci/db/connector.py b/buildbot_gentoo_ci/db/connector.py
new file mode 100644
index 0000000..682e72a
--- /dev/null
+++ b/buildbot_gentoo_ci/db/connector.py
@@ -0,0 +1,98 @@
+# This file has parts from Buildbot and is modifyed by Gentoo Authors.
+# Buildbot is free software: you can redistribute it and/or modify it
+# under the terms of the GNU General Public License as published by the
+# Free Software Foundation, version 2.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
+# details.
+#
+# You should have received a copy of the GNU General Public License along with
+# this program; if not, write to the Free Software Foundation, Inc., 51
+# Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+#
+# Copyright Buildbot Team Members
+# Origins: buildbot.db.connector.py
+# Modifyed by Gentoo Authors.
+# Copyright 2020 Gentoo Authors
+
+
+import textwrap
+
+from twisted.application import internet
+from twisted.internet import defer
+from twisted.python import log
+
+from buildbot import util
+from buildbot.db import enginestrategy
+from buildbot.db import exceptions
+from buildbot.db import pool
+from buildbot.util import service
+
+from buildbot_gentoo_ci.db import model
+
+upgrade_message = textwrap.dedent("""\
+
+ The Buildmaster database needs to be upgraded before this version of
+ buildbot can run. Use the following command-line
+
+ buildbot upgrade-master {basedir}
+
+ to upgrade the database, and try starting the buildmaster again. You may
+ want to make a backup of your buildmaster before doing so.
+ """).strip()
+
+# Gentoo Ci tables and ConnectorComponent
+class DBConnector(service.ReconfigurableServiceMixin,
+ service.AsyncMultiService):
+ # The connection between Buildbot and its backend database. This is
+ # generally accessible as master.db, but is also used during upgrades.
+ #
+ # Most of the interesting operations available via the connector are
+ # implemented in connector components, available as attributes of this
+ # object, and listed below.
+
+ # Period, in seconds, of the cleanup task. This master will perform
+ # periodic cleanup actions on this schedule.
+ CLEANUP_PERIOD = 3600
+
+ def __init__(self, basedir):
+ super().__init__()
+ self.setName('db')
+ self.basedir = basedir
+
+ # set up components
+ self._engine = None # set up in reconfigService
+ self.pool = None # set up in reconfigService
+
+ @defer.inlineCallbacks
+ def setServiceParent(self, p):
+ yield super().setServiceParent(p)
+ self.model = model.Model(self)
+
+ @defer.inlineCallbacks
+ def setup(self, config, check_version=True, verbose=True):
+ db_url = config.db['db_url']
+
+ log.msg("Setting up database with URL %r"
+ % util.stripUrlPassword(db_url))
+
+ # set up the engine and pool
+ self._engine = enginestrategy.create_engine(db_url,
+ basedir=self.basedir)
+ self.pool = pool.DBThreadPool(
+ self._engine, reactor=self.master.reactor, verbose=verbose)
+
+ # make sure the db is up to date, unless specifically asked not to
+ if check_version:
+ if db_url == 'sqlite://':
+ # Using in-memory database. Since it is reset after each process
+ # restart, `buildbot upgrade-master` cannot be used (data is not
+ # persistent). Upgrade model here to allow startup to continue.
+ self.model.upgrade()
+ current = yield self.model.is_current()
+ if not current:
+ for l in upgrade_message.format(basedir=self.basedir).split('\n'):
+ log.msg(l)
+ raise exceptions.DatabaseNotReadyError()
diff --git a/buildbot_gentoo_ci/db/migrate/README b/buildbot_gentoo_ci/db/migrate/README
new file mode 100644
index 0000000..c5f51f2
--- /dev/null
+++ b/buildbot_gentoo_ci/db/migrate/README
@@ -0,0 +1,4 @@
+This is a database migration repository.
+
+More information at
+https://sqlalchemy-migrate.readthedocs.io/en/latest/
diff --git a/buildbot_gentoo_ci/db/migrate/migrate.cfg b/buildbot_gentoo_ci/db/migrate/migrate.cfg
new file mode 100644
index 0000000..8be171d
--- /dev/null
+++ b/buildbot_gentoo_ci/db/migrate/migrate.cfg
@@ -0,0 +1,20 @@
+[db_settings]
+# Used to identify which repository this database is versioned under.
+# You can use the name of your project.
+repository_id=GentooCi
+
+# The name of the database table used to track the schema version.
+# This name shouldn't already be used by your project.
+# If this is changed once a database is under version control, you'll need to
+# change the table name in each database too.
+version_table=migrate_version
+
+# When committing a change script, Migrate will attempt to generate the
+# sql for all supported databases; normally, if one of them fails - probably
+# because you don't have that database installed - it is ignored and the
+# commit continues, perhaps ending successfully.
+# Databases in this list MUST compile successfully during a commit, or the
+# entire commit will fail. List the databases your application will actually
+# be using to ensure your updates to that database work properly.
+# This must be a list; example: ['postgres','sqlite']
+required_dbs=[]
diff --git a/buildbot_gentoo_ci/db/migrate/versions/__init__.py b/buildbot_gentoo_ci/db/migrate/versions/__init__.py
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/buildbot_gentoo_ci/db/migrate/versions/__init__.py
diff --git a/buildbot_gentoo_ci/db/model.py b/buildbot_gentoo_ci/db/model.py
new file mode 100644
index 0000000..8865517
--- /dev/null
+++ b/buildbot_gentoo_ci/db/model.py
@@ -0,0 +1,352 @@
+# This file has parts from Buildbot and is modifyed by Gentoo Authors.
+# Buildbot is free software: you can redistribute it and/or modify it
+# under the terms of the GNU General Public License as published by the
+# Free Software Foundation, version 2.
+#
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
+# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
+# details.
+#
+# You should have received a copy of the GNU General Public License along with
+# this program; if not, write to the Free Software Foundation, Inc., 51
+# Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+#
+# Copyright Buildbot Team Members
+# Origins: buildbot.db.model.py
+# Modifyed by Gentoo Authors.
+# Copyright 2020 Gentoo Authors
+
+import uuid
+import migrate
+import migrate.versioning.repository
+import sqlalchemy as sa
+from migrate import exceptions # pylint: disable=ungrouped-imports
+
+from twisted.internet import defer
+from twisted.python import log
+from twisted.python import util
+
+from buildbot.db import base
+from buildbot.db.migrate_utils import test_unicode
+from buildbot.util import sautils
+
+try:
+ from migrate.versioning.schema import ControlledSchema # pylint: disable=ungrouped-imports
+except ImportError:
+ ControlledSchema = None
+
+
+class Model(base.DBConnectorComponent):
+ #
+ # schema
+ #
+
+ metadata = sa.MetaData()
+
+ # NOTES
+
+ # * server_defaults here are included to match those added by the migration
+ # scripts, but they should not be depended on - all code accessing these
+ # tables should supply default values as necessary. The defaults are
+ # required during migration when adding non-nullable columns to existing
+ # tables.
+ #
+ # * dates are stored as unix timestamps (UTC-ish epoch time)
+ #
+ # * sqlalchemy does not handle sa.Boolean very well on MySQL or Postgres;
+ # use sa.SmallInteger instead
+
+ # Tables related to gentoo-ci-cloud
+ # -------------------------
+
+ repositorys = sautils.Table(
+ "repositorys", metadata,
+ # unique id per repository
+ sa.Column('uuid', sa.String(36), primary_key=True,
+ default=lambda: str(uuid.uuid4()),
+ ),
+ # repository's name
+ sa.Column('name', sa.String(255), nullable=False),
+ # description of the repository
+ sa.Column('description', sa.Text, nullable=True),
+ sa.Column('mirror_url', sa.String(255), nullable=True),
+ sa.Column('auto', sa.Boolean, default=False),
+ sa.Column('enabled', sa.Boolean, default=False),
+ sa.Column('ebuild', sa.Boolean, default=False),
+ )
+
+ # Use by GitPoller
+ repository_gitpuller = sautils.Table(
+ "repository_gitpuller", metadata,
+ # unique id per repository
+ sa.Column('id', sa.Integer, primary_key=True),
+ sa.Column('repository_uuid', sa.String(36),
+ sa.ForeignKey('repositorys.uuid', ondelete='CASCADE'),
+ nullable=False),
+ sa.Column('project', sa.String(255), nullable=False, default='gentoo'),
+ sa.Column('url', sa.String(255), nullable=False),
+ sa.Column('branche', sa.String(255), nullable=False, default='master'),
+ )
+
+ projects = sautils.Table(
+ "projects", metadata,
+ # unique id per project
+ sa.Column('uuid', sa.String(36), primary_key=True,
+ default=lambda: str(uuid.uuid4()),
+ ),
+ # project's name
+ sa.Column('name', sa.String(255), nullable=False),
+ # description of the project
+ sa.Column('description', sa.Text, nullable=True),
+ sa.Column('profile', sa.String(255), nullable=False),
+ sa.Column('portage_repository_uuid', sa.Integer,
+ sa.ForeignKey('repositorys.uuid', ondelete='CASCADE'),
+ nullable=False),
+ sa.Column('keyword_id', sa.Integer,
+ sa.ForeignKey('keywords.id', ondelete='CASCADE'),
+ nullable=False),
+ sa.Column('unstable', sa.Boolean, default=False),
+ sa.Column('auto', sa.Boolean, default=False),
+ sa.Column('enabled', sa.Boolean, default=False),
+ sa.Column('created_by', sa.Integer,
+ sa.ForeignKey('users.uid', ondelete='CASCADE'),
+ nullable=False),
+ )
+
+ # What repository's use by projects
+ projects_repositorys = sautils.Table(
+ "projects_repositorys", metadata,
+ sa.Column('id', sa.Integer, primary_key=True),
+ sa.Column('projects_uuid', sa.String(36),
+ sa.ForeignKey('projects.uuid', ondelete='CASCADE'),
+ nullable=False),
+ sa.Column('repository_uuid', sa.String(36),
+ sa.ForeignKey('repositorys.uuid', ondelete='CASCADE'),
+ nullable=False),
+ )
+ keywords = sautils.Table(
+ "keywords", metadata,
+ # unique id per project
+ sa.Column('id', sa.Integer, primary_key=True),
+ # project's name
+ sa.Column('keyword', sa.String(255), nullable=False),
+ )
+
+ categorys = sautils.Table(
+ "categories", metadata,
+ sa.Column('uuid', sa.String(36), primary_key=True,
+ default=lambda: str(uuid.uuid4())
+ ),
+ sa.Column('name', sa.String(255), nullable=False),
+ )
+
+ packages = sautils.Table(
+ "packages", metadata,
+ sa.Column('uuid', sa.String(36), primary_key=True,
+ default=lambda: str(uuid.uuid4()),
+ ),
+ sa.Column('name', sa.String(255), nullable=False),
+ sa.Column('category_uuid', sa.String(36),
+ sa.ForeignKey('categories.uuid', ondelete='CASCADE'),
+ nullable=False),
+ sa.Column('repository_uuid', sa.String(36),
+ sa.ForeignKey('repositorys.uuid', ondelete='CASCADE'),
+ nullable=False),
+ sa.Column('deleted', sa.Boolean, default=False),
+ sa.Column('deleted_at', sa.Integer, nullable=True),
+ )
+
+ ebuilds = sautils.Table(
+ "ebuilds", metadata,
+ sa.Column('uuid', sa.String(36), primary_key=True,
+ default=lambda: str(uuid.uuid4()),
+ ),
+ sa.Column('name', sa.String(255), nullable=False),
+ sa.Column('package_uuid', sa.String(36),
+ sa.ForeignKey('packages.uuid', ondelete='CASCADE'),
+ nullable=False),
+ sa.Column('ebuild_hash', sa.String(255), nullable=False),
+ sa.Column('deleted', sa.Boolean, default=False),
+ sa.Column('deleted_at', sa.Integer, nullable=True),
+ )
+
+ ebuildkeywords = sautils.Table(
+ "ebuildkeywords", metadata,
+ # unique id per project
+ sa.Column('id', sa.Integer, primary_key=True),
+ # project's name
+ sa.Column('keyword_id', sa.Integer,
+ sa.ForeignKey('keywords.id', ondelete='CASCADE')),
+ sa.Column('ebuild_uuid', sa.String(36),
+ sa.ForeignKey('ebuilds.uuid', ondelete='CASCADE')),
+ sa.Column('status', sa.String(255), nullable=False),
+ )
+
+ # Tables related to users
+ # -----------------------
+
+ # This table identifies individual users, and contains buildbot-specific
+ # information about those users.
+ users = sautils.Table(
+ "users", metadata,
+ # unique user id number
+ sa.Column("uid", sa.Integer, primary_key=True),
+
+ # identifier (nickname) for this user; used for display
+ sa.Column("identifier", sa.String(255), nullable=False),
+
+ # username portion of user credentials for authentication
+ sa.Column("bb_username", sa.String(128)),
+
+ # password portion of user credentials for authentication
+ sa.Column("bb_password", sa.String(128)),
+ )
+
+ # Indexes
+ # -------
+
+
+
+ # MySQL creates indexes for foreign keys, and these appear in the
+ # reflection. This is a list of (table, index) names that should be
+ # expected on this platform
+
+ implied_indexes = [
+ ]
+
+ # Migration support
+ # -----------------
+
+ # this is a bit more complicated than might be expected because the first
+ # seven database versions were once implemented using a homespun migration
+ # system, and we need to support upgrading masters from that system. The
+ # old system used a 'version' table, where SQLAlchemy-Migrate uses
+ # 'migrate_version'
+
+ repo_path = util.sibpath(__file__, "migrate")
+
+ @defer.inlineCallbacks
+ def is_current(self):
+ if ControlledSchema is None:
+ # this should have been caught earlier by enginestrategy.py with a
+ # nicer error message
+ raise ImportError("SQLAlchemy/SQLAlchemy-Migrate version conflict")
+
+ def thd(engine):
+ # we don't even have to look at the old version table - if there's
+ # no migrate_version, then we're not up to date.
+ repo = migrate.versioning.repository.Repository(self.repo_path)
+ repo_version = repo.latest
+ try:
+ # migrate.api doesn't let us hand in an engine
+ schema = ControlledSchema(engine, self.repo_path)
+ db_version = schema.version
+ except exceptions.DatabaseNotControlledError:
+ return False
+
+ return db_version == repo_version
+ ret = yield self.db.pool.do_with_engine(thd)
+ return ret
+
+ # returns a Deferred that returns None
+ def create(self):
+ # this is nice and simple, but used only for tests
+ def thd(engine):
+ self.metadata.create_all(bind=engine)
+ return self.db.pool.do_with_engine(thd)
+
+ @defer.inlineCallbacks
+ def upgrade(self):
+
+ # here, things are a little tricky. If we have a 'version' table, then
+ # we need to version_control the database with the proper version
+ # number, drop 'version', and then upgrade. If we have no 'version'
+ # table and no 'migrate_version' table, then we need to version_control
+ # the database. Otherwise, we just need to upgrade it.
+
+ def table_exists(engine, tbl):
+ try:
+ r = engine.execute("select * from {} limit 1".format(tbl))
+ r.close()
+ return True
+ except Exception:
+ return False
+
+ # http://code.google.com/p/sqlalchemy-migrate/issues/detail?id=100
+ # means we cannot use the migrate.versioning.api module. So these
+ # methods perform similar wrapping functions to what is done by the API
+ # functions, but without disposing of the engine.
+ def upgrade(engine):
+ schema = ControlledSchema(engine, self.repo_path)
+ changeset = schema.changeset(None)
+ with sautils.withoutSqliteForeignKeys(engine):
+ for version, change in changeset:
+ log.msg('migrating schema version {} -> {}'.format(version, version + 1))
+ schema.runchange(version, change, 1)
+
+ def check_sqlalchemy_migrate_version():
+ # sqlalchemy-migrate started including a version number in 0.7; we
+ # support back to 0.6.1, but not 0.6. We'll use some discovered
+ # differences between 0.6.1 and 0.6 to get that resolution.
+ version = getattr(migrate, '__version__', 'old')
+ if version == 'old':
+ try:
+ from migrate.versioning import schemadiff
+ if hasattr(schemadiff, 'ColDiff'):
+ version = "0.6.1"
+ else:
+ version = "0.6"
+ except Exception:
+ version = "0.0"
+ version_tup = tuple(map(int, version.split('-', 1)[0].split('.')))
+ log.msg("using SQLAlchemy-Migrate version {}".format(version))
+ if version_tup < (0, 6, 1):
+ raise RuntimeError(("You are using SQLAlchemy-Migrate {}. "
+ "The minimum version is 0.6.1.").format(version))
+
+ def version_control(engine, version=None):
+ ControlledSchema.create(engine, self.repo_path, version)
+
+ # the upgrade process must run in a db thread
+ def thd(engine):
+ # if the migrate_version table exists, we can just let migrate
+ # take care of this process.
+ if table_exists(engine, 'migrate_version'):
+ r = engine.execute(
+ "select version from migrate_version limit 1")
+ old_version = r.scalar()
+ if old_version < 40:
+ raise EightUpgradeError()
+ try:
+ upgrade(engine)
+ except sa.exc.NoSuchTableError as e: # pragma: no cover
+ if 'migration_tmp' in str(e):
+ log.err('A serious error has been encountered during the upgrade. The '
+ 'previous upgrade has been likely interrupted. The database has '
+ 'been damaged and automatic recovery is impossible.')
+ log.err('If you believe this is an error, please submit a bug to the '
+ 'Buildbot project.')
+ raise
+
+ # if the version table exists, then we can version_control things
+ # at that version, drop the version table, and let migrate take
+ # care of the rest.
+ elif table_exists(engine, 'version'):
+ raise EightUpgradeError()
+
+ # otherwise, this db is new, so we don't bother using the migration engine
+ # and just create the tables, and put the version directly to
+ # latest
+ else:
+ # do some tests before getting started
+ test_unicode(engine)
+
+ log.msg("Initializing empty database")
+ Model.metadata.create_all(engine)
+ repo = migrate.versioning.repository.Repository(self.repo_path)
+
+ version_control(engine, repo.latest)
+
+ check_sqlalchemy_migrate_version()
+ yield self.db.pool.do_with_engine(thd)