~launchpad-pqm/launchpad/devel

9678.20.2 by Max Bowsher
Here a shebang, there a shebang, everywhere a shebang!
1
#!/usr/bin/python2.5
8452.3.3 by Karl Fogel
* utilities/: Add copyright header block to source files that were
2
#
8687.15.2 by Karl Fogel
In files modified by r8688, change "<YEARS>" to "2009", as per
3
# Copyright 2009 Canonical Ltd.  This software is licensed under the
8687.15.3 by Karl Fogel
Shorten the copyright header block to two lines.
4
# GNU Affero General Public License version 3 (see the file LICENSE).
8452.3.3 by Karl Fogel
* utilities/: Add copyright header block to source files that were
5
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
6
"""
7
Restore a full database dump. This script should become unnecessary
8
when we are running PostgreSQL 8 as it will correctly order its dumps.
9
"""
10
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
11
import sys, os, tempfile
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
12
from optparse import OptionParser
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
13
from subprocess import Popen, PIPE
14
from signal import SIGTERM
15
16
class DumpFile(object):
17
    """File-like object wrapping a normal or compressed file on disk"""
18
    process = None
19
20
    def __init__(self, dumpname):
21
22
        if dumpname.endswith('.bz2'):
23
            self.process = Popen(
24
                    ["bunzip2", "-c", dumpname], stdin=PIPE, stdout=PIPE
25
                    )
26
        elif dumpname.endswith('.gz'):
27
            self.process = Popen(
28
                    ["gunzip", "-c", dumpname], stdin=PIPE, stdout=PIPE
29
                    )
30
        else:
31
            self.out = open(dumpname, 'r')
32
33
        if self.process is not None:
34
            self.process.stdin.close()
35
            self.out = self.process.stdout
36
37
    def read(self, bytes=None):
38
        return self.out.read(bytes)
39
40
    def close(self):
41
        if self.process is None:
42
            return self.out.close()
43
44
        if self.process.poll() is not None:
45
            if self.process.returncode != 0:
46
                print >> sys.stderr, "ERROR: Uncompressor returned %d" % (
47
                        self.process.returncode
48
                        )
49
            return
50
51
        os.kill(self.process.pid, SIGTERM)
52
53
    def fileno(self):
54
        return self.out.fileno()
55
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
56
57
def generate_order(dumpname):
58
    """Generate a correctly order dump listing"""
59
   
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
60
    dump_input = DumpFile(dumpname)
61
    cmd = Popen('pg_restore -l', shell=True, stdout=PIPE, stdin=dump_input)
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
62
    (stdout, stderr) = cmd.communicate()
63
    if cmd.returncode != 0:
64
        raise RuntimeError('pg_restore returned %d' % rv)
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
65
    dump_input.close()
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
66
67
    full_listing = [l for l in stdout.split('\n') if l.strip()]
68
69
    full_listing.sort(listing_cmp)
70
71
    return '\n'.join(full_listing)
72
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
73
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
74
def listing_cmp(a, b):
2936.1.1 by Stuart Bishop
Update pgrestore.py to resture PostgreSQL 7.4 dumps under PostgreSQL 8.0
75
    if POSTGRESQL7:
76
        idx = 2
77
    else:
78
        idx = 3
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
79
    if a.startswith(';'):
80
        atype = ';'
81
    else:
2936.1.1 by Stuart Bishop
Update pgrestore.py to resture PostgreSQL 7.4 dumps under PostgreSQL 8.0
82
        atype = a.split()[idx]
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
83
84
    if b.startswith(';'):
85
        btype = ';'
86
    else:
2936.1.1 by Stuart Bishop
Update pgrestore.py to resture PostgreSQL 7.4 dumps under PostgreSQL 8.0
87
        btype = b.split()[idx]
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
88
2936.1.3 by Stuart Bishop
pgrestore helper should create INDEXes earlier so they can be used by contraints checks
89
    # Build indexes in same parse as tables, so that tables with constraints
90
    # that need to reference other tables (eg. CHECK is_person() columns)
91
    # load in a reasonable time. This is more fragile, but should last
92
    # until postgresql 8 migration makes this script irrelevant.
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
93
    scores = {
94
        ';': 0,
95
        'SCHEMA': 1,
96
        'TYPE': 10,
97
        'FUNC': 10,
98
        'PROCEDURAL': 10,
99
        'FUNCTION': 10,
100
        'OPERATOR': 20,
101
        'TABLE': 30,
102
        'SEQUENCE': 35,
103
        'BLOBS': 38,
104
        'VIEW': 40,
105
        'TRIGGER': 90,
106
        'FK': 95,
107
        'CONSTRAINT': 95,
2936.1.3 by Stuart Bishop
pgrestore helper should create INDEXes earlier so they can be used by contraints checks
108
        'INDEX': 30,
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
109
        'COMMENT': 200,
110
        'ACL': 1000,
111
        }
112
113
    # Will fail if we get an unknown type in the listing, which is by design
114
    # at the moment. Might want a default value though instead?
115
    return cmp(scores[atype], scores[btype])
116
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
117
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
118
def createdb(options):
119
    args = ['createdb', '--encoding=UNICODE']
120
121
    if options.user:
122
        args.append('--username=%s' % options.user)
123
124
    if options.host:
125
        args.append('--host=%s' % options.host)
126
127
    args.append(options.dbname)
128
129
    if options.verbose:
130
        cmd = ' '.join(args)
131
        print >> sys.stderr, 'Running %s' % cmd
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
132
        createdb = Popen(args)
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
133
    else:
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
134
        createdb = Popen(args, stdout=PIPE, stderr=PIPE)
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
135
136
    (out, err) = createdb.communicate()
137
138
    if createdb.returncode != 0:
139
        print >> sys.stderr, err
140
        print >> sys.stderr, out
141
        print >> sys.stderr, 'ERROR: %d' % createdb.returncode
142
        sys.exit(createdb.returncode)
143
144
145
if __name__ == "__main__":
146
147
    parser = OptionParser("usage: %prog [options] [DUMPFILE | -]")
148
    parser.add_option(
149
            "-d", "--dbname", dest="dbname",
150
            help="Create the database DBNAME and restore the dump into it",
151
            metavar="DBNAME"
152
            )
153
    parser.add_option(
154
            "-H", "--host", dest="host", default=None,
155
            help="Connect to PostgreSQL running on HOST",
156
            metavar="HOST"
157
            )
158
    parser.add_option(
159
            "--no-acl", dest="perms", action="store_false",
160
            default=True, help="Do not restore ownership or permissions",
161
            )
162
    parser.add_option(
163
            "-U", "--user", dest="user",
164
            help="Connect as superuser USER", metavar="USER", default=None
165
            )
166
    parser.add_option(
167
            "-v", "--verbose", dest="verbose",
168
            action="store_true", default=False
169
            )
2936.1.1 by Stuart Bishop
Update pgrestore.py to resture PostgreSQL 7.4 dumps under PostgreSQL 8.0
170
    parser.add_option(
171
            "-7", dest="postgres7",
172
            action="store_true", default=False,
173
            help="Restore into a PostgreSQL 7.4 database"
174
            )
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
175
176
    (options, args) = parser.parse_args()
177
2936.1.1 by Stuart Bishop
Update pgrestore.py to resture PostgreSQL 7.4 dumps under PostgreSQL 8.0
178
    if options.postgres7:
179
        POSTGRESQL7 = True
180
    else:
181
        POSTGRESQL7 = False
182
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
183
    if len(args) > 1:
184
        parser.error("Too many arguments")
185
186
    if len(args) == 0:
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
187
        parser.error("Must specify dump file name.")
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
188
189
    dumpname = args[0]
190
191
    if not os.path.exists(dumpname):
192
        parser.error("%s does not exist" % dumpname)
193
194
    handle, listingfilename = tempfile.mkstemp()
195
    try:
196
        os.close(handle)
197
        listingfile = open(listingfilename, 'w')
198
        listingfile.write(generate_order(dumpname))
199
        listingfile.close()
200
201
        pg_restore_args = ["pg_restore", "--use-list=%s" % listingfilename]
202
203
        if options.dbname:
204
            createdb(options)
205
            pg_restore_args.append("--dbname=%s" % options.dbname)
206
207
        if not options.perms:
208
            pg_restore_args.append("--no-owner")
209
            pg_restore_args.append("--no-acl")
210
211
        if options.user:
212
            pg_restore_args.append("--user=%s" % options.user)
213
214
        if options.host:
215
            pg_restore_args.append("--host=%s" % options.host)
216
217
        if options.verbose:
218
            pg_restore_args.append("--verbose")
219
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
220
        dump_input = DumpFile(dumpname)
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
221
222
        if options.verbose:
223
            cmd = ' '.join(pg_restore_args)
224
            print >> sys.stderr, "Running %s" % cmd
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
225
            rest = Popen(pg_restore_args, stdin=dump_input)
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
226
        else:
1876 by Canonical.com Patch Queue Manager
[r=jamesh] database dump script
227
            rest = Popen(pg_restore_args, stderr=PIPE, stdin=dump_input)
1745 by Canonical.com Patch Queue Manager
[r=jamesh] Simple database restoration script and sampledata fixes
228
229
        (out,err) = rest.communicate()
230
        if rest.returncode != 0:
231
            print >> sys.stderr, err
232
            print >> sys.stderr, 'ERROR: %d' % rest.returncode
233
            sys.exit(rest.returncode)
234
235
    finally:
236
        os.unlink(listingfilename)
237
238