~launchpad-pqm/launchpad/devel

10637.3.1 by Guilherme Salgado
Use the default python version instead of a hard-coded version
1
#!/usr/bin/python -S
8687.15.22 by Karl Fogel
Add the copyright header block to the remaining .py files.
2
#
3
# Copyright 2009 Canonical Ltd.  This software is licensed under the
4
# GNU Affero General Public License version 3 (see the file LICENSE).
3691.361.1 by Stuart Bishop
Initial cut at account closure script
5
5516.1.4 by Mark Shuttleworth
Tests for r=gmb
6
"""Remove personal details of a user from the database, leaving a stub."""
3691.361.1 by Stuart Bishop
Initial cut at account closure script
7
8
__metaclass__ = type
9
__all__ = []
10
11
import _pythonpath
12
13
from optparse import OptionParser
14
import sys
15
6198.1.62 by Stuart Bishop
More review feedback
16
from canonical.database.sqlbase import connect, sqlvalues
3691.361.1 by Stuart Bishop
Initial cut at account closure script
17
from canonical.launchpad.scripts import db_options, logger_options, logger
12915.5.6 by Curtis Hovey
Fixed script import.
18
from lp.answers.enums import QuestionStatus
11882.2.2 by Jonathan Lange
Clear up a heck of a lot of imports from canonical.launchpad.interfaces.
19
from lp.registry.interfaces.person import PersonCreationRationale
20
3691.361.1 by Stuart Bishop
Initial cut at account closure script
21
22
def close_account(con, log, username):
23
    """Close a person's account.
5516.1.4 by Mark Shuttleworth
Tests for r=gmb
24
3691.361.1 by Stuart Bishop
Initial cut at account closure script
25
    Return True on success, or log an error message and return False
26
    """
27
    cur = con.cursor()
28
    cur.execute("""
7675.395.80 by Stuart Bishop
Fix scripts/close-account.py
29
        SELECT Person.id, Person.account, name, teamowner
3691.361.2 by Stuart Bishop
Allow email address on command line and tidy
30
        FROM Person LEFT OUTER JOIN EmailAddress
31
            ON Person.id = EmailAddress.person
32
        WHERE name=%(username)s or lower(email)=lower(%(username)s)
3691.361.1 by Stuart Bishop
Initial cut at account closure script
33
        """, vars())
34
    try:
7675.395.80 by Stuart Bishop
Fix scripts/close-account.py
35
        person_id, account_id, username, teamowner = cur.fetchone()
3691.361.1 by Stuart Bishop
Initial cut at account closure script
36
    except TypeError:
37
        log.fatal("User %s does not exist" % username)
38
        return False
39
40
    # We don't do teams
41
    if teamowner is not None:
42
        log.fatal("%s is a team" % username)
43
        return False
44
3691.361.2 by Stuart Bishop
Allow email address on command line and tidy
45
    log.info("Closing %s's account" % username)
46
3691.361.1 by Stuart Bishop
Initial cut at account closure script
47
    def table_notification(table):
48
        log.debug("Handling the %s table" % table)
49
50
    # All names starting with 'removed' are blacklisted, so this will always
51
    # succeed.
52
    new_name = 'removed%d' % person_id
53
7675.395.80 by Stuart Bishop
Fix scripts/close-account.py
54
    # Remove the EmailAddress. This is the most important step, as
55
    # people requesting account removal seem to primarily be interested
56
    # in ensuring we no longer store this information.
57
    table_notification('EmailAddress')
6198.1.15 by Stuart Bishop
Fix close-account.py and tests
58
    cur.execute("""
7675.395.80 by Stuart Bishop
Fix scripts/close-account.py
59
        DELETE FROM EmailAddress WHERE person = %s
6198.1.62 by Stuart Bishop
More review feedback
60
        """ % sqlvalues(person_id))
61
3691.361.1 by Stuart Bishop
Initial cut at account closure script
62
    # Clean out personal details from the Person table
63
    table_notification('Person')
64
    unknown_rationale = PersonCreationRationale.UNKNOWN.value
65
    cur.execute("""
66
        UPDATE Person
6198.1.15 by Stuart Bishop
Fix close-account.py and tests
67
        SET displayname='Removed by request',
6198.1.62 by Stuart Bishop
More review feedback
68
            name=%(new_name)s, language=NULL, account=NULL,
7675.749.8 by Stuart Bishop
Fix account closure script
69
            homepage_content=NULL, icon=NULL, mugshot=NULL,
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
70
            hide_email_addresses=TRUE, registrant=NULL, logo=NULL,
3691.361.1 by Stuart Bishop
Initial cut at account closure script
71
            creation_rationale=%(unknown_rationale)s, creation_comment=NULL
72
        WHERE id=%(person_id)s
73
        """, vars())
74
7675.395.80 by Stuart Bishop
Fix scripts/close-account.py
75
    # Remove the Account. We don't set the status to deactivated,
76
    # as this script is used to satisfy people who insist on us removing
77
    # all their personal details from our systems. This includes any
78
    # identification tokens like email addresses or openid identifiers.
79
    # So the Account record would be unusable, and contain no useful
80
    # information.
81
    table_notification('Account')
82
    if account_id is not None:
83
        cur.execute("""
84
            DELETE FROM Account WHERE id = %s
85
            """ % sqlvalues(account_id))
86
3691.361.1 by Stuart Bishop
Initial cut at account closure script
87
    # Reassign their bugs
88
    table_notification('BugTask')
89
    cur.execute("""
90
        UPDATE BugTask SET assignee=NULL WHERE assignee=%(person_id)s
91
        """, vars())
92
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
93
    # Reassign questions assigned to the user, and close all their questions
3691.361.1 by Stuart Bishop
Initial cut at account closure script
94
    # since nobody else can
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
95
    table_notification('Question')
3691.361.1 by Stuart Bishop
Initial cut at account closure script
96
    cur.execute("""
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
97
        UPDATE Question SET assignee=NULL WHERE assignee=%(person_id)s
3691.361.1 by Stuart Bishop
Initial cut at account closure script
98
        """, vars())
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
99
    closed_question_status = QuestionStatus.SOLVED.value
3691.361.1 by Stuart Bishop
Initial cut at account closure script
100
    cur.execute("""
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
101
        UPDATE Question
102
        SET status=%(closed_question_status)s, whiteboard=
3691.361.1 by Stuart Bishop
Initial cut at account closure script
103
            'Closed by Launchpad due to owner requesting account removal'
104
        WHERE owner=%(person_id)s
105
        """, vars())
106
107
    # Remove rows from tables in simple cases in the given order
108
    removals = [
109
        # Trash their email addresses. Unsociable privacy nut jobs who request
110
        # account removal would be pissed if they reregistered with their old
111
        # email address and this resurrected their deleted account, as the
112
        # email address is probably the piece of data we store that they where
113
        # most concerned with being removed from our systems.
114
        ('EmailAddress', 'person'),
115
116
        # Trash their codes of conduct and GPG keys
117
        ('SignedCodeOfConduct', 'owner'),
118
        ('GpgKey', 'owner'),
119
120
        # Subscriptions
121
        ('BranchSubscription', 'person'),
122
        ('BugSubscription', 'person'),
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
123
        ('QuestionSubscription', 'person'),
3691.361.1 by Stuart Bishop
Initial cut at account closure script
124
        ('POSubscription', 'person'),
125
        ('SpecificationSubscription', 'person'),
126
127
        # Personal stuff, freeing up the namespace for others who want to play
128
        # or just to remove any fingerprints identifying the user.
129
        ('IrcId', 'person'),
130
        ('JabberId', 'person'),
131
        ('WikiName', 'person'),
132
        ('PersonLanguage', 'person'),
5516.1.1 by Mark Shuttleworth
Initial cut of db for location
133
        ('PersonLocation', 'person'),
3691.361.1 by Stuart Bishop
Initial cut at account closure script
134
        ('SshKey', 'person'),
10303.1.1 by Gary Poster
use newest version of zc.buildout
135
3691.361.1 by Stuart Bishop
Initial cut at account closure script
136
        # Karma
137
        ('Karma', 'person'),
138
        ('KarmaCache', 'person'),
139
        ('KarmaTotalCache', 'person'),
140
141
        # Team memberships
142
        ('TeamMembership', 'person'),
143
        ('TeamParticipation', 'person'),
144
145
        # Contacts
6025.3.27 by Tom Berger
delete from PackageBugSupervisor when deleting an account
146
        ('PackageBugSupervisor', 'bug_supervisor'),
4360.1.2 by Stuart Bishop
Fix close-account.py and add a basic test
147
        ('AnswerContact', 'person'),
3691.361.1 by Stuart Bishop
Initial cut at account closure script
148
149
        # Pending items in queues
150
        ('POExportRequest', 'person'),
151
152
        # Access lists
153
        ('PushMirrorAccess', 'person'),
154
        ('DistroComponentUploader', 'uploader'),
155
        ]
156
    for table, person_id_column in removals:
157
        table_notification(table)
158
        cur.execute("""
159
                DELETE FROM %(table)s WHERE %(person_id_column)s=%(person_id)d
160
                """ % vars())
161
162
    # Trash Sprint Attendance records in the future.
163
    table_notification('SprintAttendance')
164
    cur.execute("""
165
        DELETE FROM SprintAttendance
166
        USING Sprint
167
        WHERE Sprint.id = SprintAttendance.sprint
168
            AND attendee=%(person_id)s
169
            AND Sprint.time_starts > CURRENT_TIMESTAMP AT TIME ZONE 'UTC'
170
        """, vars())
171
172
    return True
173
174
def main():
3691.361.2 by Stuart Bishop
Allow email address on command line and tidy
175
    parser = OptionParser(
176
            '%prog [options] (username|email) [...]'
177
            )
3691.361.1 by Stuart Bishop
Initial cut at account closure script
178
    db_options(parser)
179
    logger_options(parser)
180
181
    (options, args) = parser.parse_args()
182
183
    if len(args) == 0:
184
        parser.error("Must specify username (Person.name)")
185
186
    log = logger(options)
187
3691.361.2 by Stuart Bishop
Allow email address on command line and tidy
188
    con = None
3691.361.1 by Stuart Bishop
Initial cut at account closure script
189
    try:
190
        log.debug("Connecting to database")
191
        con = connect(options.dbuser)
192
        for username in args:
193
            if not close_account(con, log, username):
194
                log.debug("Rolling back")
195
                con.rollback()
196
                return 1
197
        log.debug("Committing changes")
198
        con.commit()
199
        return 0
200
    except:
201
        log.exception("Unhandled exception")
202
        log.debug("Rolling back")
3691.361.2 by Stuart Bishop
Allow email address on command line and tidy
203
        if con is not None:
204
            con.rollback()
3691.361.1 by Stuart Bishop
Initial cut at account closure script
205
        return 1
206
207
if __name__ == '__main__':
208
    sys.exit(main())