##// END OF EJS Templates
Imported some of the GPLv3'd changes from RhodeCode v2.2.5....
Imported some of the GPLv3'd changes from RhodeCode v2.2.5. This imports changes between changesets 21af6c4eab3d and 6177597791c2 in RhodeCode's original repository, including only changes to Python files and HTML. RhodeCode clearly licensed its changes to these files under GPLv3 in their /LICENSE file, which states the following: The Python code and integrated HTML are licensed under the GPLv3 license. (See: https://code.rhodecode.com/rhodecode/files/v2.2.5/LICENSE or http://web.archive.org/web/20140512193334/https://code.rhodecode.com/rhodecode/files/f3b123159901f15426d18e3dc395e8369f70ebe0/LICENSE for an online copy of that LICENSE file) Conservancy reviewed these changes and confirmed that they can be licensed as a whole to the Kallithea project under GPLv3-only. While some of the contents committed herein are clearly licensed GPLv3-or-later, on the whole we must assume the are GPLv3-only, since the statement above from RhodeCode indicates that they intend GPLv3-only as their license, per GPLv3ยง14 and other relevant sections of GPLv3.

File last commit:

r3886:a1696507 beta
r4116:ffd45b18 rhodecode-2.2.5-gpl
Show More
genmodel.py
284 lines | 10.5 KiB | text/x-python | PythonLexer
added dbmigrate package, added model changes...
r833 """
update migrations for 1.2
r1442 Code to generate a Python model from a database or differences
between a model and database.
added dbmigrate package, added model changes...
r833
update migrations for 1.2
r1442 Some of this is borrowed heavily from the AutoCode project at:
http://code.google.com/p/sqlautocode/
added dbmigrate package, added model changes...
r833 """
import sys
import logging
import sqlalchemy
fixed imports on migrate, added getting current version from database
r835 from rhodecode.lib.dbmigrate import migrate
from rhodecode.lib.dbmigrate.migrate import changeset
added dbmigrate package, added model changes...
r833
update migrations for 1.2
r1442
added dbmigrate package, added model changes...
r833 log = logging.getLogger(__name__)
HEADER = """
## File autogenerated by genmodel.py
from sqlalchemy import *
meta = MetaData()
"""
DECLARATIVE_HEADER = """
## File autogenerated by genmodel.py
from sqlalchemy import *
from sqlalchemy.ext import declarative
Base = declarative.declarative_base()
"""
class ModelGenerator(object):
update migrations for 1.2
r1442 """Various transformations from an A, B diff.
In the implementation, A tends to be called the model and B
the database (although this is not true of all diffs).
The diff is directionless, but transformations apply the diff
in a particular direction, described in the method name.
"""
added dbmigrate package, added model changes...
r833
def __init__(self, diff, engine, declarative=False):
self.diff = diff
self.engine = engine
self.declarative = declarative
def column_repr(self, col):
kwarg = []
if col.key != col.name:
kwarg.append('key')
if col.primary_key:
col.primary_key = True # otherwise it dumps it as 1
kwarg.append('primary_key')
if not col.nullable:
kwarg.append('nullable')
if col.onupdate:
kwarg.append('onupdate')
if col.default:
if col.primary_key:
# I found that PostgreSQL automatically creates a
# default value for the sequence, but let's not show
# that.
pass
else:
kwarg.append('default')
update migrations for 1.2
r1442 args = ['%s=%r' % (k, getattr(col, k)) for k in kwarg]
added dbmigrate package, added model changes...
r833
# crs: not sure if this is good idea, but it gets rid of extra
# u''
name = col.name.encode('utf8')
type_ = col.type
for cls in col.type.__class__.__mro__:
if cls.__module__ == 'sqlalchemy.types' and \
not cls.__name__.isupper():
if cls is not type_.__class__:
type_ = cls()
break
update migrations for 1.2
r1442 type_repr = repr(type_)
if type_repr.endswith('()'):
type_repr = type_repr[:-2]
constraints = [repr(cn) for cn in col.constraints]
added dbmigrate package, added model changes...
r833 data = {
'name': name,
update migrations for 1.2
r1442 'commonStuff': ', '.join([type_repr] + constraints + args),
}
added dbmigrate package, added model changes...
r833
update migrations for 1.2
r1442 if self.declarative:
return """%(name)s = Column(%(commonStuff)s)""" % data
added dbmigrate package, added model changes...
r833 else:
update migrations for 1.2
r1442 return """Column(%(name)r, %(commonStuff)s)""" % data
added dbmigrate package, added model changes...
r833
update migrations for 1.2
r1442 def _getTableDefn(self, table, metaName='meta'):
added dbmigrate package, added model changes...
r833 out = []
tableName = table.name
if self.declarative:
out.append("class %(table)s(Base):" % {'table': tableName})
update migrations for 1.2
r1442 out.append(" __tablename__ = '%(table)s'\n" %
{'table': tableName})
added dbmigrate package, added model changes...
r833 for col in table.columns:
update migrations for 1.2
r1442 out.append(" %s" % self.column_repr(col))
out.append('\n')
added dbmigrate package, added model changes...
r833 else:
update migrations for 1.2
r1442 out.append("%(table)s = Table('%(table)s', %(meta)s," %
{'table': tableName, 'meta': metaName})
added dbmigrate package, added model changes...
r833 for col in table.columns:
update migrations for 1.2
r1442 out.append(" %s," % self.column_repr(col))
out.append(")\n")
added dbmigrate package, added model changes...
r833 return out
updated sqlalchemy migrate to latest version
r1061 def _get_tables(self,missingA=False,missingB=False,modified=False):
added dbmigrate package, added model changes...
r833 to_process = []
updated sqlalchemy migrate to latest version
r1061 for bool_,names,metadata in (
(missingA,self.diff.tables_missing_from_A,self.diff.metadataB),
(missingB,self.diff.tables_missing_from_B,self.diff.metadataA),
(modified,self.diff.tables_different,self.diff.metadataA),
added dbmigrate package, added model changes...
r833 ):
if bool_:
for name in names:
yield metadata.tables.get(name)
fixed imports on migrate, added getting current version from database
r835
update migrations for 1.2
r1442 def genBDefinition(self):
"""Generates the source code for a definition of B.
Assumes a diff where A is empty.
Was: toPython. Assume database (B) is current and model (A) is empty.
"""
added dbmigrate package, added model changes...
r833 out = []
if self.declarative:
out.append(DECLARATIVE_HEADER)
else:
out.append(HEADER)
out.append("")
for table in self._get_tables(missingA=True):
update migrations for 1.2
r1442 out.extend(self._getTableDefn(table))
added dbmigrate package, added model changes...
r833 return '\n'.join(out)
update migrations for 1.2
r1442 def genB2AMigration(self, indent=' '):
use consisten double quote docstring formatting
r3886 """Generate a migration from B to A.
update migrations for 1.2
r1442
Was: toUpgradeDowngradePython
Assume model (A) is most current and database (B) is out-of-date.
use consisten double quote docstring formatting
r3886 """
update migrations for 1.2
r1442
decls = ['from migrate.changeset import schema',
'pre_meta = MetaData()',
'post_meta = MetaData()',
]
upgradeCommands = ['pre_meta.bind = migrate_engine',
'post_meta.bind = migrate_engine']
downgradeCommands = list(upgradeCommands)
for tn in self.diff.tables_missing_from_A:
pre_table = self.diff.metadataB.tables[tn]
decls.extend(self._getTableDefn(pre_table, metaName='pre_meta'))
upgradeCommands.append(
"pre_meta.tables[%(table)r].drop()" % {'table': tn})
downgradeCommands.append(
"pre_meta.tables[%(table)r].create()" % {'table': tn})
added dbmigrate package, added model changes...
r833
update migrations for 1.2
r1442 for tn in self.diff.tables_missing_from_B:
post_table = self.diff.metadataA.tables[tn]
decls.extend(self._getTableDefn(post_table, metaName='post_meta'))
upgradeCommands.append(
"post_meta.tables[%(table)r].create()" % {'table': tn})
downgradeCommands.append(
"post_meta.tables[%(table)r].drop()" % {'table': tn})
added dbmigrate package, added model changes...
r833
update migrations for 1.2
r1442 for (tn, td) in self.diff.tables_different.iteritems():
if td.columns_missing_from_A or td.columns_different:
pre_table = self.diff.metadataB.tables[tn]
decls.extend(self._getTableDefn(
pre_table, metaName='pre_meta'))
if td.columns_missing_from_B or td.columns_different:
post_table = self.diff.metadataA.tables[tn]
decls.extend(self._getTableDefn(
post_table, metaName='post_meta'))
for col in td.columns_missing_from_A:
upgradeCommands.append(
'pre_meta.tables[%r].columns[%r].drop()' % (tn, col))
downgradeCommands.append(
'pre_meta.tables[%r].columns[%r].create()' % (tn, col))
for col in td.columns_missing_from_B:
upgradeCommands.append(
'post_meta.tables[%r].columns[%r].create()' % (tn, col))
downgradeCommands.append(
'post_meta.tables[%r].columns[%r].drop()' % (tn, col))
for modelCol, databaseCol, modelDecl, databaseDecl in td.columns_different:
added dbmigrate package, added model changes...
r833 upgradeCommands.append(
updated sqlalchemy migrate to latest version
r1061 'assert False, "Can\'t alter columns: %s:%s=>%s"' % (
update migrations for 1.2
r1442 tn, modelCol.name, databaseCol.name))
added dbmigrate package, added model changes...
r833 downgradeCommands.append(
updated sqlalchemy migrate to latest version
r1061 'assert False, "Can\'t alter columns: %s:%s=>%s"' % (
update migrations for 1.2
r1442 tn, modelCol.name, databaseCol.name))
added dbmigrate package, added model changes...
r833
return (
'\n'.join(decls),
update migrations for 1.2
r1442 '\n'.join('%s%s' % (indent, line) for line in upgradeCommands),
'\n'.join('%s%s' % (indent, line) for line in downgradeCommands))
added dbmigrate package, added model changes...
r833
updated sqlalchemy migrate to latest version
r1061 def _db_can_handle_this_change(self,td):
update migrations for 1.2
r1442 """Check if the database can handle going from B to A."""
added dbmigrate package, added model changes...
r833 if (td.columns_missing_from_B
and not td.columns_missing_from_A
and not td.columns_different):
update migrations for 1.2
r1442 # Even sqlite can handle column additions.
added dbmigrate package, added model changes...
r833 return True
else:
return not self.engine.url.drivername.startswith('sqlite')
update migrations for 1.2
r1442 def runB2A(self):
"""Goes from B to A.
Was: applyModel. Apply model (A) to current database (B).
"""
added dbmigrate package, added model changes...
r833
meta = sqlalchemy.MetaData(self.engine)
for table in self._get_tables(missingA=True):
table = table.tometadata(meta)
table.drop()
for table in self._get_tables(missingB=True):
table = table.tometadata(meta)
table.create()
for modelTable in self._get_tables(modified=True):
tableName = modelTable.name
modelTable = modelTable.tometadata(meta)
dbTable = self.diff.metadataB.tables[tableName]
td = self.diff.tables_different[tableName]
source code cleanup: remove trailing white space, normalize file endings
r1203
added dbmigrate package, added model changes...
r833 if self._db_can_handle_this_change(td):
source code cleanup: remove trailing white space, normalize file endings
r1203
added dbmigrate package, added model changes...
r833 for col in td.columns_missing_from_B:
modelTable.columns[col].create()
for col in td.columns_missing_from_A:
dbTable.columns[col].drop()
# XXX handle column changes here.
else:
# Sqlite doesn't support drop column, so you have to
# do more: create temp table, copy data to it, drop
# old table, create new table, copy data back.
#
# I wonder if this is guaranteed to be unique?
tempName = '_temp_%s' % modelTable.name
def getCopyStatement():
preparer = self.engine.dialect.preparer
commonCols = []
for modelCol in modelTable.columns:
if modelCol.name in dbTable.columns:
commonCols.append(modelCol.name)
commonColsStr = ', '.join(commonCols)
return 'INSERT INTO %s (%s) SELECT %s FROM %s' % \
(tableName, commonColsStr, commonColsStr, tempName)
# Move the data in one transaction, so that we don't
# leave the database in a nasty state.
connection = self.engine.connect()
trans = connection.begin()
try:
connection.execute(
'CREATE TEMPORARY TABLE %s as SELECT * from %s' % \
(tempName, modelTable.name))
# make sure the drop takes place inside our
# transaction with the bind parameter
modelTable.drop(bind=connection)
modelTable.create(bind=connection)
connection.execute(getCopyStatement())
connection.execute('DROP TABLE %s' % tempName)
trans.commit()
except:
trans.rollback()
raise