File: sqlite2pg

package info (click to toggle)
trac 1.2.3%2Bdfsg-1
  • links: PTS, VCS
  • area: main
  • in suites: buster
  • size: 25,004 kB
  • sloc: python: 75,314; makefile: 450; sh: 79; xml: 10
file content (336 lines) | stat: -rw-r--r-- 12,081 bytes parent folder | download | duplicates (8)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
#!/usr/bin/env python

# -*- coding: utf-8 -*-
#
# Copyright (C) 2008 John Hampton <pacopablo@pacopablo.com>
# All rights reserved.
#
# This software is licensed as described in the file COPYING, which
# you should have received as part of this distribution. The terms
# are also available at http://trac.edgewall.com/license.html.
#
# This software consists of voluntary contributions made by many
# individuals. For the exact contribution history, see the revision
# history and logs, available at:
# http://trac-hacks.org/wiki/SqliteToPgScript
#
# Basically, it boils down to: feel free to use/modify/distribute/etc.
# However, give me credit where due.  Also, if you like the script and
# find it useful, buy me a soda or candy bar or something if ever we
# meet. Thanks and enjoy.
#
# Author: John Hampton <pacopablo@pacopablo.com>


import os, os.path, sys
from trac.env import Environment, EnvironmentSetup, IEnvironmentSetupParticipant
from trac.core import ComponentMeta
from trac.db import DatabaseManager
from optparse import OptionParser
from psycopg2 import ProgrammingError, IntegrityError
from psycopg2.extensions import ISOLATION_LEVEL_AUTOCOMMIT

VERSION='0.10'

class TableMigration(object):
    """
        Class to conatin all table migration functions
    """
    def __init__(self, sqlenv, pgenv, opts):
        """ Create a TableMigration instance.  Required are SQLite and
            PostgreSQL environment objects
        """
        self.sqlenv = sqlenv
        self.pgenv = pgenv
        self.opts = opts
        self.sdb = self.sqlenv.get_db_cnx()
        self.pgdb = self.pgenv.get_db_cnx()
        self.pgdb_schema = self.pgdb.schema
        pass

    def lookupMethod(self, table):
        """ Get a reference to the function that handles migration for the
            specified table
        """
        m = getattr(self, ''.join(['migrate_', table.upper()]), None)
        if not m:
            m = self.default_copy
        return m

    def cleanTable(self, table):
        """ Clear the contents of the table """
        cur = self.pgdb.cursor()
        delete_from = "DELETE FROM %s" % table
        cur.execute(delete_from)

    def migrateTable(self, table):
        """ Migrate the table specified. """
        if not self.opts.noclean:
            self.cleanTable(table)
        m = getattr(self, ''.join(['migrate_', table.upper()]), None)
        if not m:
            rc = self.default_copy(table)
        else:
            rc = m()
        return rc

    def default_copy(self, table):
        """ Copy the table from the sqlite db to the postgresql db """
        select_all = "SELECT * FROM %s" % table
        scur = self.sdb.cursor()
        pgcur = self.pgdb.cursor()
        scur.execute(select_all)
        cols = scur.description
        if not cols:
            return True
        subs = ["%s" for x in range(len(cols))]
        insert_into = ''.join([ "INSERT INTO ", table, " VALUES (",
                                ','.join(subs), ")"])
        rows = row_exists = 0
        for row in scur:
            rows += 1
            try:
                pgcur.execute(insert_into, row)
            except (ProgrammingError, IntegrityError):
                row_exists += 1
            continue
        if row_exists:
            print "%s of %s rows already existed in the %s table" % \
                  (str(row_exists), str(rows), table)
        self.pgdb.commit()
        return row_exists > 0

    def migrate_TICKET(self):
        """ Migrate the ticket table and adjust the sequences properly """
        self.default_copy('ticket')
        select_maxticket = "SELECT max(id) FROM ticket"
        pgcur = self.pgdb.cursor()
        pgcur.execute(select_maxticket)
        r = pgcur.fetchone()
        if r:
            pgcur.execute("SELECT setval('ticket_id_seq', %s)", r)
        self.pgdb.commit()

    def migrate_REPORT(self):
        """ Migrate the report table and adjust the sequences properly """
        self.default_copy('report')
        select_maxreport = "SELECT max(id) FROM report"
        pgcur = self.pgdb.cursor()
        pgcur.execute(select_maxreport)
        r = pgcur.fetchone()
        if r:
            pgcur.execute("SELECT setval('report_id_seq', %s)", r)
        self.pgdb.commit()

    def migrate_PERMISSION(self):
        """
            Migrate permission table
        """
        scur = self.sdb.cursor()
        pgcur = self.pgdb.cursor()
        rows = row_exists = 0
        if not self.opts.plist:
            sql_select = "SELECT * FROM permission"
        else:
            subs = []
            for x in range(len(self.opts.plist)):
                subs.append("%s")
                continue
            sql_select = ''.join([  "SELECT * FROM permission ",
                                    "WHERE username NOT IN (",
                                    ','.join(subs),
                                    ")",
                                 ])

        scur.execute(sql_select, self.opts.plist)
        sql_insert = "INSERT INTO permission VALUES (%s, %s)"
        for row in scur:
            rows += 1
            try:
                pgcur.execute(sql_insert, row)
            except (ProgrammingError, IntegrityError):
                row_exists += 1
            continue
        if row_exists:
            print "%s of %s rows already existed in the permission table" % (str(row_exists), str(rows))
        self.pgdb.commit()
        return row_exists > 0

    def migrate_WIKI(self):
        """
            Migrate wiki table
        """
        scur = self.sdb.cursor()
        pgcur = self.pgdb.cursor()
        rows = row_exists = 0
        if not self.opts.wlist:
            sql_select = "SELECT * FROM wiki"
        else:
            subs = []
            for x in range(len(self.opts.wlist)):
                subs.append("%s")
                continue
            sql_select = ''.join([  "SELECT * FROM wiki ",
                                    "WHERE name NOT IN (",
                                    ','.join(subs),
                                    ")",
                                 ])

        scur.execute(sql_select, self.opts.wlist)
        sql_insert = "INSERT INTO wiki VALUES (%s, %s, %s, %s, %s, %s, %s, %s)"
        for row in scur:
            rows += 1
            try:
                pgcur.execute(sql_insert, row)
            except (ProgrammingError, IntegrityError):
                row_exists += 1
            continue
        if row_exists:
            print "%s of %s rows already existed in the wiki table" % (str(row_exists), str(rows))
        self.pgdb.commit()
        return row_exists > 0


def getSQLiteEnvironment(opts):
    """ Create an Environment connected to the SQLite database """

    dburi = opts.sqlite_uri
    env = Environment(opts.tracenv)
    env.config.set('trac', 'database', dburi)
    return env

def getPostgreSQLEnvironment(opts):
    """ Create an Environment connected to the PostgreSQL database """

    dburi = opts.pg_uri
    env = Environment(opts.tracenv)
    env.config.set('trac', 'database', dburi)
    try:
        cnx = env.get_db_cnx()
        cur = cnx.cursor()
        cur.execute("select value from system where name = 'database_version'");
    except ProgrammingError:
        cnx.rollback()
        DatabaseManager(env).init_db()
        DatabaseManager(env).shutdown()
        for x in filter(None, [env.compmgr[cls] for cls in
                        ComponentMeta._registry.get(
                        IEnvironmentSetupParticipant, [])]):
            if isinstance(x, EnvironmentSetup):
                x.environment_created()
    if env.needs_upgrade():
        env.upgrade()
    return env

def getAllTables(env):
    """ Queries the PostgreSQL database for a list of tables """
    cnx = env.get_db_cnx()
    schema = cnx.schema or u'public'
    cur = cnx.cursor()
    select_tables = """SELECT tablename
                         FROM pg_catalog.pg_tables
                        WHERE schemaname = %s"""
    cur.execute(select_tables, (schema,))
    cnx.commit()
    return [table[0] for table in cur]

def Main(opts):
    """
        Migrate data from SQLite to PostgreSQL
    """
    rc = 0

    sqlenv = getSQLiteEnvironment(opts)
    pgenv = getPostgreSQLEnvironment(opts)
    tmigration = TableMigration(sqlenv, pgenv, opts)
    if not opts.tlist:
        opts.tlist = getAllTables(pgenv)
    for tname in opts.tlist:
        try:
            rc = tmigration.migrateTable(tname) or rc
        except AttributeError:
            print "Migration of %s has not been implemented" % tname
            pass
        continue

    return rc

def doArgs(argv):
    """ Look if you can't guess what this function does, just give up now. """
    global VERSION

    version = "%%prog %s" % VERSION
    usage ="usage: %prog [options] [site]"
    description="%prog is used to migrate data from SQLite to PostgreSQL."

    parser = OptionParser(usage=usage, version=version, description=description)

    parser.add_option("-t", "--tracbase", dest="tracbase", type="string",
                        help="Parent path for trac sites",
                        metavar="<path>")
    parser.add_option("-e", "--tracenv", dest="tracenv", type="string",
                        help="Path to trac environment",
                        metavar="<path>")
    parser.add_option("-m", "--migrate", dest="migrate", type="string",
                        help="Comma separated list of tables to migrate",
                        metavar="<list>", default=None)
    parser.add_option("", "--exclude_perms", dest="perms_exclude",
                        type="string", help="Comma separated list of users to "
                        "exclude from permission migration", metavar="<list>")
    parser.add_option("-w", "--wikipages", dest="wikipages", type="string",
                        help="Comma separated list of wiki page names to "
                        "ommit from the migration", metavar="<list>")
    parser.add_option("-p", "--pg_uri", dest="pg_uri", type="string",
                        help="DB URI for PostgreSQL database",
                        metavar="<uri>")
    parser.add_option("-s", "--sqlite_uri", dest="sqlite_uri", type="string",
                        help="DB URI for SQLite database",
                        metavar="<uri>", default="sqlite:db/trac.db")
    parser.add_option("", "--noclean", dest="noclean", action="store_true",
                        help="Do not clear PostgreSQL tables before transfer",
                        default=False)

    (options, args) = parser.parse_args(argv)
    if not options.tracenv:
        if  not options.tracbase:
            print ("You must specify the --tracenv or the --tracbase option")
            sys.exit(1)
        else:
            if len(args) < 1:
                print ("You must specify a project name\n")
                sys.exit(1)
            options.project = args[0]
            options.tracenv = os.path.join(options.tracbase, args[0])

    if not options.pg_uri or not options.pg_uri.startswith('postgres://'):
        print ("You must specify a valid URI for the PostgreSQL database.")
        print ("  eg. postgres://user:password@localhost/dbname")
        sys.exit(1)

    if not options.sqlite_uri or not options.sqlite_uri.startswith('sqlite:'):
        print ("You must specify a valid URI for the SQLite database.")
        print ("  eg. sqlite:db/trac.db")
        sys.exit(1)


    options.args = args
    options.tlist = options.migrate and \
                    [t.strip() for t in options.migrate.strip().split(',')]
    options.wlist = options.wikipages and \
                    [w.strip() for w in options.wikipages.strip().split(',')] \
                    or []
    options.plist = options.perms_exclude and \
                    [p.strip() for p in options.perms_exclude.strip().split(',')] \
                    or []

    return options


def main(argv):
    opts = doArgs(argv)
    Main(opts)
    return 0

if __name__ == '__main__':
    sys.exit(main(sys.argv[1:]))