Skip to content

Commit

Permalink
Drop the updates.title column.
Browse files Browse the repository at this point in the history
The updates.title field used to be a space separated list of the
builds found in the update, with a uniqueness constraint. There
were at least two problems with it.

One problem was that the list of builds was not consistently
updated (fedora-infra#1946) and sometimes the title did not reflect the current
set of builds in the update. This was more severe than it may seem,
because it turns out that the CLI was using the title to set the
list of builds when editing updates, instead of using the list of
builds. This means that the CLI could set the list of builds back
to a former set, rather than leaving it at the set it was
currently at. Note that the CLI does not currently support editing
the set of builds on an update (fedora-infra#3037), so it is especially
surprising when this happens.

The second problem is that large updates with many builds end up
generating a very long title and PostgreSQL raises an error
because it refuses to index strings beyond a certain length. It
was found, for example, that release engineering could not create
an update with 300 builds in it. Since we plan to use Bodhi to work
with side tags as part of gating Fedora Rawhide, it will be
important for users to be able to create updates with large
numbers of builds.

The title was also not a particularly useful field. It was possible
to use it as a key to access an update, but that was a poor use for
it since the title changes when the list of builds change, and
because the order of the list of builds was important. This led to
unpredictable and unstable URLs for updates, and Bodhi already had
an update alias field which is stable and is better suited for that
purpose.

This commit drops the field from the database and removes it from
being used to reference updates in the API. It preserves it in API
responses, however, but now generates it dynamically so that the
title is always an accurate list of the builds included in the
update.

fixes fedora-infra#186
fixes fedora-infra#1542
fixes fedora-infra#1714
fixes fedora-infra#1946

Signed-off-by: Randy Barlow <[email protected]>
  • Loading branch information
bowlofeggs committed Mar 13, 2019
1 parent bc1685e commit fb329d0
Show file tree
Hide file tree
Showing 40 changed files with 818 additions and 1,104 deletions.
43 changes: 17 additions & 26 deletions bodhi/client/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -420,7 +420,7 @@ def _validate_edit_update(ctx, param, value):
"""
Validate the update argument given to the updates edit command.
The update argument can only be update id or update title
The update argument can only be an update id.
Args:
param (basestring): The name of the parameter being validated. Unused.
Expand All @@ -430,11 +430,10 @@ def _validate_edit_update(ctx, param, value):
Raises:
click.BadParameter: If the value is invalid.
"""
if re.search(bindings.UPDATE_ID_RE, value)\
or re.search(bindings.UPDATE_TITLE_RE, value):
if re.search(bindings.UPDATE_ID_RE, value):
return value
else:
raise click.BadParameter("Please provide an Update ID or an Update Title")
raise click.BadParameter("Please provide an Update ID")


@updates.command()
Expand All @@ -451,11 +450,11 @@ def edit(user, password, url, debug, openid_api, **kwargs):
"""
Edit an existing update.
UPDATE: The title of the update (e.g. FEDORA-2017-f8e0ef2850)
UPDATE: The alias of the update (e.g. FEDORA-2017-f8e0ef2850)
"""
# Developer Docs
"""
The update argument can be an update id or the update title.
The update argument must be an update id.
Args:
user (unicode): The username to authenticate as.
Expand All @@ -472,26 +471,19 @@ def edit(user, password, url, debug, openid_api, **kwargs):
kwargs['notes'] = _get_notes(**kwargs)

try:
if re.search(bindings.UPDATE_ID_RE, kwargs['update']):
query_param = {'updateid': kwargs['update']}
resp = client.query(**query_param)
title = resp['updates'][0]['title']
else:
# _validate_edit_update() has already ensured that we either got an update ID or an NVR,
# so we can assume here that we have an NVR.
query_param = {'like': kwargs['update']}
resp = client.query(**query_param)
title = kwargs['update']
query_param = {'updateid': kwargs['update']}
resp = client.query(**query_param)
del(kwargs['update'])
kwargs['builds'] = title
kwargs['edited'] = title

# Convert list of 'Bug' instances in DB to comma separated bug_ids for parsing.
former_update = resp['updates'][0].copy()
if not kwargs['bugs']:
kwargs['bugs'] = ",".join([str(bug['bug_id']) for bug in former_update['bugs']])
former_update.pop('bugs', None)

kwargs['builds'] = [b['nvr'] for b in former_update['builds']]
kwargs['edited'] = former_update['alias']

# Replace empty fields with former values from database.
for field in kwargs:
if kwargs[field] in (None, '') and field in former_update:
Expand All @@ -507,7 +499,6 @@ def edit(user, password, url, debug, openid_api, **kwargs):

@updates.command()
@click.option('--updateid', help='Query by update ID (eg: FEDORA-2015-0001)')
@click.option('--title', help='Query by title')
@click.option('--alias', help='Query by alias')
@click.option('--approved-since', help='Approved after a specific timestamp')
@click.option('--approved-before', help='Approved before a specific timestamp')
Expand Down Expand Up @@ -593,7 +584,7 @@ def request(update, state, user, password, url, openid_api, **kwargs):
"""
Change an update's request status.
UPDATE: The title of the update (e.g. FEDORA-2017-f8e0ef2850)
UPDATE: The alias of the update (e.g. FEDORA-2017-f8e0ef2850)
STATE: The state you wish to change the update's request to. Valid options are
testing, stable, obsolete, unpush, and revoke.
Expand Down Expand Up @@ -641,7 +632,7 @@ def comment(update, text, karma, user, password, url, openid_api, **kwargs):
"""
Comment on an update.
UPDATE: The title of the update (e.g. FEDORA-2017-f8e0ef2850)
UPDATE: The alias of the update (e.g. FEDORA-2017-f8e0ef2850)
TEXT: the comment to be added to the update
"""
Expand Down Expand Up @@ -719,7 +710,7 @@ def download(url, **kwargs):
# *think* that should ever be possible for these opts.

for update in resp.updates:
click.echo("Downloading packages from {0}".format(update['title']))
click.echo("Downloading packages from {0}".format(update['alias']))
for build in update['builds']:
# subprocess is icky, but koji module doesn't
# expose this in any usable way, and we don't want
Expand Down Expand Up @@ -783,13 +774,13 @@ def waive(update, show, test, comment, url, openid_api, **kwargs):
"""
Show or waive unsatified requirements (ie: missing or failing tests) on an existing update.
UPDATE: The title of the update (e.g. FEDORA-2017-f8e0ef2850)
UPDATE: The alias of the update (e.g. FEDORA-2017-f8e0ef2850)
COMMENT: A comment explaining why the requirements were waived (mandatory with --test)
"""
# Developer Docs
"""
The update argument can be an update id or the update title.
The update argument must be an update id..
Args:
update (unicode): The update who unsatisfied requirements wish to waive.
Expand Down Expand Up @@ -1016,7 +1007,7 @@ def print_resp(resp, client, verbose=False, override_hint=True):
resp.total, len(resp.updates)))
elif resp.get('update'):
click.echo(client.update_str(resp['update']))
elif 'title' in resp:
elif resp.get('alias'):
click.echo(client.update_str(resp))
elif 'overrides' in resp:
if len(resp.overrides) == 1:
Expand All @@ -1034,7 +1025,7 @@ def print_resp(resp, client, verbose=False, override_hint=True):
if override_hint:
_print_override_koji_hint(resp, client)
elif 'comment' in resp:
click.echo('The following comment was added to %s' % resp.comment['update'].title)
click.echo('The following comment was added to %s' % resp.comment['update'].alias)
click.echo(resp.comment.text)
elif 'compose' in resp:
click.echo(client.compose_str(resp['compose'], minimal=False))
Expand Down
31 changes: 11 additions & 20 deletions bodhi/client/bindings.py
Original file line number Diff line number Diff line change
Expand Up @@ -224,7 +224,7 @@ def save(self, **kwargs):
Save an update.
This entails either creating a new update, or editing an existing one.
To edit an existing update, you must specify the update title in
To edit an existing update, you must specify the update alias in
the ``edited`` keyword argument.
Args:
Expand All @@ -248,7 +248,7 @@ def save(self, **kwargs):
stable_karma (int): The upper threshold for marking an update as
``stable``.
unstable_karma (int): The lower threshold for unpushing an update.
edited (basestring): The update title of the existing update that we are
edited (basestring): The update alias of the existing update that we are
editing.
severity (basestring): The severity of this update (``urgent``, ``high``,
``medium``, ``low``).
Expand All @@ -275,7 +275,7 @@ def request(self, update, request):
Request an update state change.
Args:
update (basestring): The title of the update.
update (basestring): The alias of the update.
request (basestring): The request (``testing``, ``stable``, ``obsolete``,
``unpush``, ``revoke``).
Returns:
Expand All @@ -301,7 +301,7 @@ def waive(self, update, comment, tests=None):
Waive unsatisfied requirements on an update.
Args:
update (basestring): The title of the update.
update (basestring): The alias of the update.
comment (basestring): A comment explaining the waiver.
tests (tuple(basestring) or None): The list of unsatisfied requirements
to waive. If not specified, all unsatisfied requirements of this
Expand All @@ -328,7 +328,6 @@ def query(self, **kwargs):
Query bodhi for a list of updates.
Args:
title (basestring): The update title.
alias (basestring): The update alias.
updateid (basestring): The update ID (eg: FEDORA-2015-0001).
content_type (basestring): A content type (rpm, module) to limit the query to.
Expand Down Expand Up @@ -374,9 +373,6 @@ def query(self, **kwargs):
Returns:
munch.Munch: The response from Bodhi describing the query results.
"""
if 'title' in kwargs:
kwargs['like'] = kwargs['title']
del kwargs['title']
# bodhi1 compat
if 'limit' in kwargs:
kwargs['rows_per_page'] = kwargs['limit']
Expand Down Expand Up @@ -414,7 +410,7 @@ def get_test_status(self, update):
Query bodhi for the test status of the specified update..
Args:
update (basestring): The title or identifier of the update to
update (basestring): The alias of the update to
retrieve the test status of.
Returns:
munch.Munch: The response from Bodhi describing the query results.
Expand All @@ -427,7 +423,7 @@ def comment(self, update, comment, karma=0, email=None):
Add a comment to an update.
Args:
update (basestring): The title of the update comment on.
update (basestring): The alias of the update comment on.
comment (basestring): The text of the comment to add to the update.
karma (int): The amount of karma to leave. May be -1, 0, or 1. Defaults to 0.
email (basestring or None): Email address for an anonymous user. If an email address is
Expand Down Expand Up @@ -796,16 +792,15 @@ def update_str(self, update, minimal=False):
update_lines = ['{:=^80}\n'.format('=')]
update_lines += [
line + '\n' for line in textwrap.wrap(
update['title'].replace(',', ', '),
update['title'],
width=80,
initial_indent=' ' * 5,
subsequent_indent=' ' * 5)
]
update_lines.append('{:=^80}\n'.format('='))

if update['alias']:
update_lines.append(
line_formatter.format('Update ID', update['alias']))
update_lines.append(
line_formatter.format('Update ID', update['alias']))

update_lines += [
line_formatter.format('Content Type', update['content_type']),
Expand Down Expand Up @@ -900,12 +895,8 @@ def update_str(self, update, minimal=False):
comments_lines[1:])
]

if update['alias']:
update_lines.append(
'\n {0}updates/{1}\n'.format(self.base_url, update['alias']))
else:
update_lines.append(
'\n {0}updates/{1}\n'.format(self.base_url, update['title']))
update_lines.append(
'\n {0}updates/{1}\n'.format(self.base_url, update['alias']))

return ''.join(update_lines)

Expand Down
16 changes: 4 additions & 12 deletions bodhi/server/consumers/composer.py
Original file line number Diff line number Diff line change
Expand Up @@ -211,14 +211,6 @@ def _get_composes(self, msg):
with self.db_factory() as db:
if 'api_version' in msg and msg['api_version'] == 2:
composes = [Compose.from_dict(db, c) for c in msg['composes']]
elif 'updates' in msg:
updates = [db.query(Update).filter(Update.title == t).one() for t in msg['updates']]
composes = Compose.from_updates(updates)
for c in composes:
db.add(c)
# This flush is necessary so the compose finds its updates, which gives it a
# content_type when it is serialized later.
db.flush()
else:
raise ValueError('Unable to process fedmsg: {}'.format(msg))

Expand Down Expand Up @@ -384,7 +376,7 @@ def work(self):
notifications.publish(
topic="compose.composing",
msg=dict(repo=self.id,
updates=[u.title for u in self.compose.updates],
updates=[' '.join([b.nvr for b in u.builds]) for u in self.compose.updates],
agent=self.agent,
ctype=self.ctype.value),
force=True,
Expand Down Expand Up @@ -470,7 +462,7 @@ def perform_gating(self):
for update in self.compose.updates:
result, reason = update.check_requirements(self.db, config)
if not result:
self.log.warning("%s failed gating: %s" % (update.title, reason))
self.log.warning("%s failed gating: %s" % (update.alias, reason))
self.eject_from_compose(update, reason)
# We may have removed some updates from this compose above, and do we don't want future
# reads on self.compose.updates to see those, so let's mark that attribute expired so
Expand All @@ -487,7 +479,7 @@ def eject_from_compose(self, update, reason):
comment on the update, in a log message, and in a fedmsg.
"""
update.locked = False
text = '%s ejected from the push because %r' % (update.title, reason)
text = '%s ejected from the push because %r' % (update.alias, reason)
log.warning(text)
update.comment(self.db, text, author=u'bodhi')
# Remove the pending tag as well
Expand Down Expand Up @@ -722,7 +714,7 @@ def modify_bugs(self):
"""Mark bugs on each Update as modified."""
self.log.info('Updating bugs')
for update in self.compose.updates:
self.log.debug('Modifying bugs for %s', update.title)
self.log.debug('Modifying bugs for %s', update.alias)
update.modify_bugs()

@checkpoint
Expand Down
9 changes: 2 additions & 7 deletions bodhi/server/consumers/updates.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
# Copyright 2015-2018 Red Hat Inc., and others.
# Copyright 2015-2019 Red Hat Inc., and others.
#
# This file is part of Bodhi.
#
Expand Down Expand Up @@ -111,11 +111,6 @@ def consume(self, message):
# https://github.com/fedora-infra/bodhi/issues/458
time.sleep(1)

if not alias:
log.error("Update Handler got update with no "
"alias %s." % pprint.pformat(msg))
return

with self.db_factory() as session:
update = Update.get(alias)
if not update:
Expand Down Expand Up @@ -202,7 +197,7 @@ def work_on_bugs(self, session, update, bugs):

log.info("Commenting on %r" % bug.bug_id)
comment = config['initial_bug_msg'] % (
update.title, update.release.long_name, update.abs_url())
update.alias, update.release.long_name, update.abs_url())

log.info("Modifying %r" % bug.bug_id)
bug.modified(update, comment)
Expand Down
2 changes: 0 additions & 2 deletions bodhi/server/metadata.py
Original file line number Diff line number Diff line change
Expand Up @@ -146,8 +146,6 @@ def __init__(self, release, request, db, composedir, close_shelf=True):

self.uinfo = cr.UpdateInfo()
for update in self.updates:
if not update.alias:
update.assign_alias()
self.add_update(update)

if close_shelf:
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
# Copyright (c) 2019 Red Hat, Inc.
#
# This file is part of Bodhi.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
"""
Remove the updates.title column.
Revision ID: 58b7919b942c
Revises: aae0d29d49b7
Create Date: 2019-02-20 17:13:01.260748
"""
from alembic import op
import sqlalchemy as sa


# revision identifiers, used by Alembic.
revision = '58b7919b942c'
down_revision = 'eec610d7ab3a'


def upgrade():
"""Drop the updates.title column."""
op.alter_column('updates', 'alias', existing_type=sa.BOOLEAN(), nullable=False)
op.drop_index('ix_updates_title', table_name='updates')
op.drop_column('updates', 'title')


def downgrade():
"""Bring back the updates.title column and try to guess what it should be set to."""
op.add_column('updates', sa.Column('title', sa.TEXT(), autoincrement=False, nullable=True))
# Set the title back to something similar to what it might have been. This isn't guaranteed to
# be the title it was before, because we can't know what order the nvrs appeared in, but it will
# set it to the expected format and expected set of NVRs. Single build updates should at least
# get their old title back.
op.execute(
("UPDATE updates SET title=("
"SELECT string_agg(nvr, ' ') as title FROM ("
"SELECT builds.nvr FROM builds WHERE update_id=updates.id ORDER BY nvr) as nvr)"))
op.create_index('ix_updates_title', 'updates', ['title'], unique=True)
op.alter_column('updates', 'alias', existing_type=sa.BOOLEAN(), nullable=True)
Loading

0 comments on commit fb329d0

Please sign in to comment.