|
@@
-34,6
+34,8
b' from sqlalchemy.ext.hybrid import hybrid'
|
|
34
|
34
|
from sqlalchemy.orm import relationship, joinedload, class_mapper, validates
|
|
35
|
35
|
from beaker.cache import cache_region, region_invalidate
|
|
36
|
36
|
|
|
|
37
|
_ = lambda s: s
|
|
|
38
|
|
|
37
|
39
|
from rhodecode.lib.vcs import get_backend
|
|
38
|
40
|
from rhodecode.lib.vcs.utils.helpers import get_scm
|
|
39
|
41
|
from rhodecode.lib.vcs.exceptions import VCSError
|
|
@@
-156,7
+158,7
b' class RhodeCodeSetting(Base, BaseModel):'
|
|
156
|
158
|
__tablename__ = 'rhodecode_settings'
|
|
157
|
159
|
__table_args__ = (
|
|
158
|
160
|
UniqueConstraint('app_settings_name'),
|
|
159
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
161
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
160
|
162
|
'mysql_charset': 'utf8'}
|
|
161
|
163
|
)
|
|
162
|
164
|
app_settings_id = Column("app_settings_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-231,7
+233,7
b' class RhodeCodeUi(Base, BaseModel):'
|
|
231
|
233
|
__tablename__ = 'rhodecode_ui'
|
|
232
|
234
|
__table_args__ = (
|
|
233
|
235
|
UniqueConstraint('ui_key'),
|
|
234
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
236
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
235
|
237
|
'mysql_charset': 'utf8'}
|
|
236
|
238
|
)
|
|
237
|
239
|
|
|
@@
-282,7
+284,7
b' class User(Base, BaseModel):'
|
|
282
|
284
|
__tablename__ = 'users'
|
|
283
|
285
|
__table_args__ = (
|
|
284
|
286
|
UniqueConstraint('username'), UniqueConstraint('email'),
|
|
285
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
287
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
286
|
288
|
'mysql_charset': 'utf8'}
|
|
287
|
289
|
)
|
|
288
|
290
|
user_id = Column("user_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-404,7
+406,7
b' class User(Base, BaseModel):'
|
|
404
|
406
|
class UserLog(Base, BaseModel):
|
|
405
|
407
|
__tablename__ = 'user_logs'
|
|
406
|
408
|
__table_args__ = (
|
|
407
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
409
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
408
|
410
|
'mysql_charset': 'utf8'},
|
|
409
|
411
|
)
|
|
410
|
412
|
user_log_id = Column("user_log_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-426,7
+428,7
b' class UserLog(Base, BaseModel):'
|
|
426
|
428
|
class UsersGroup(Base, BaseModel):
|
|
427
|
429
|
__tablename__ = 'users_groups'
|
|
428
|
430
|
__table_args__ = (
|
|
429
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
431
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
430
|
432
|
'mysql_charset': 'utf8'},
|
|
431
|
433
|
)
|
|
432
|
434
|
|
|
@@
-468,7
+470,7
b' class UsersGroup(Base, BaseModel):'
|
|
468
|
470
|
class UsersGroupMember(Base, BaseModel):
|
|
469
|
471
|
__tablename__ = 'users_groups_members'
|
|
470
|
472
|
__table_args__ = (
|
|
471
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
473
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
472
|
474
|
'mysql_charset': 'utf8'},
|
|
473
|
475
|
)
|
|
474
|
476
|
|
|
@@
-488,7
+490,7
b' class Repository(Base, BaseModel):'
|
|
488
|
490
|
__tablename__ = 'repositories'
|
|
489
|
491
|
__table_args__ = (
|
|
490
|
492
|
UniqueConstraint('repo_name'),
|
|
491
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
493
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
492
|
494
|
'mysql_charset': 'utf8'},
|
|
493
|
495
|
)
|
|
494
|
496
|
|
|
@@
-675,6
+677,23
b' class Repository(Base, BaseModel):'
|
|
675
|
677
|
grouped[cmt.revision].append(cmt)
|
|
676
|
678
|
return grouped
|
|
677
|
679
|
|
|
|
680
|
def statuses(self, revisions=None):
|
|
|
681
|
"""
|
|
|
682
|
Returns statuses for this repository
|
|
|
683
|
:param revisions: list of revisions to get statuses for
|
|
|
684
|
:type revisions: list
|
|
|
685
|
"""
|
|
|
686
|
|
|
|
687
|
statuses = ChangesetStatus.query()\
|
|
|
688
|
.filter(ChangesetStatus.repo == self)
|
|
|
689
|
if revisions:
|
|
|
690
|
statuses = statuses.filter(ChangesetStatus.revision.in_(revisions))
|
|
|
691
|
grouped = defaultdict(list)
|
|
|
692
|
for stat in statuses.all():
|
|
|
693
|
grouped[statuses.revision].append(stat)
|
|
|
694
|
return grouped
|
|
|
695
|
|
|
|
696
|
|
|
678
|
697
|
#==========================================================================
|
|
679
|
698
|
# SCM CACHE INSTANCE
|
|
680
|
699
|
#==========================================================================
|
|
@@
-738,7
+757,7
b' class RepoGroup(Base, BaseModel):'
|
|
738
|
757
|
__table_args__ = (
|
|
739
|
758
|
UniqueConstraint('group_name', 'group_parent_id'),
|
|
740
|
759
|
CheckConstraint('group_id != group_parent_id'),
|
|
741
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
760
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
742
|
761
|
'mysql_charset': 'utf8'},
|
|
743
|
762
|
)
|
|
744
|
763
|
__mapper_args__ = {'order_by': 'group_name'}
|
|
@@
-867,7
+886,7
b' class RepoGroup(Base, BaseModel):'
|
|
867
|
886
|
class Permission(Base, BaseModel):
|
|
868
|
887
|
__tablename__ = 'permissions'
|
|
869
|
888
|
__table_args__ = (
|
|
870
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
889
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
871
|
890
|
'mysql_charset': 'utf8'},
|
|
872
|
891
|
)
|
|
873
|
892
|
permission_id = Column("permission_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-906,7
+925,7
b' class UserRepoToPerm(Base, BaseModel):'
|
|
906
|
925
|
__tablename__ = 'repo_to_perm'
|
|
907
|
926
|
__table_args__ = (
|
|
908
|
927
|
UniqueConstraint('user_id', 'repository_id', 'permission_id'),
|
|
909
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
928
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
910
|
929
|
'mysql_charset': 'utf8'}
|
|
911
|
930
|
)
|
|
912
|
931
|
repo_to_perm_id = Column("repo_to_perm_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-935,7
+954,7
b' class UserToPerm(Base, BaseModel):'
|
|
935
|
954
|
__tablename__ = 'user_to_perm'
|
|
936
|
955
|
__table_args__ = (
|
|
937
|
956
|
UniqueConstraint('user_id', 'permission_id'),
|
|
938
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
957
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
939
|
958
|
'mysql_charset': 'utf8'}
|
|
940
|
959
|
)
|
|
941
|
960
|
user_to_perm_id = Column("user_to_perm_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-950,7
+969,7
b' class UsersGroupRepoToPerm(Base, BaseMod'
|
|
950
|
969
|
__tablename__ = 'users_group_repo_to_perm'
|
|
951
|
970
|
__table_args__ = (
|
|
952
|
971
|
UniqueConstraint('repository_id', 'users_group_id', 'permission_id'),
|
|
953
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
972
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
954
|
973
|
'mysql_charset': 'utf8'}
|
|
955
|
974
|
)
|
|
956
|
975
|
users_group_to_perm_id = Column("users_group_to_perm_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-979,7
+998,7
b' class UsersGroupToPerm(Base, BaseModel):'
|
|
979
|
998
|
__tablename__ = 'users_group_to_perm'
|
|
980
|
999
|
__table_args__ = (
|
|
981
|
1000
|
UniqueConstraint('users_group_id', 'permission_id',),
|
|
982
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1001
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
983
|
1002
|
'mysql_charset': 'utf8'}
|
|
984
|
1003
|
)
|
|
985
|
1004
|
users_group_to_perm_id = Column("users_group_to_perm_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-994,7
+1013,7
b' class UserRepoGroupToPerm(Base, BaseMode'
|
|
994
|
1013
|
__tablename__ = 'user_repo_group_to_perm'
|
|
995
|
1014
|
__table_args__ = (
|
|
996
|
1015
|
UniqueConstraint('user_id', 'group_id', 'permission_id'),
|
|
997
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1016
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
998
|
1017
|
'mysql_charset': 'utf8'}
|
|
999
|
1018
|
)
|
|
1000
|
1019
|
|
|
@@
-1012,7
+1031,7
b' class UsersGroupRepoGroupToPerm(Base, Ba'
|
|
1012
|
1031
|
__tablename__ = 'users_group_repo_group_to_perm'
|
|
1013
|
1032
|
__table_args__ = (
|
|
1014
|
1033
|
UniqueConstraint('users_group_id', 'group_id'),
|
|
1015
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1034
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1016
|
1035
|
'mysql_charset': 'utf8'}
|
|
1017
|
1036
|
)
|
|
1018
|
1037
|
|
|
@@
-1030,7
+1049,7
b' class Statistics(Base, BaseModel):'
|
|
1030
|
1049
|
__tablename__ = 'statistics'
|
|
1031
|
1050
|
__table_args__ = (
|
|
1032
|
1051
|
UniqueConstraint('repository_id'),
|
|
1033
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1052
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1034
|
1053
|
'mysql_charset': 'utf8'}
|
|
1035
|
1054
|
)
|
|
1036
|
1055
|
stat_id = Column("stat_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-1048,7
+1067,7
b' class UserFollowing(Base, BaseModel):'
|
|
1048
|
1067
|
__table_args__ = (
|
|
1049
|
1068
|
UniqueConstraint('user_id', 'follows_repository_id'),
|
|
1050
|
1069
|
UniqueConstraint('user_id', 'follows_user_id'),
|
|
1051
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1070
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1052
|
1071
|
'mysql_charset': 'utf8'}
|
|
1053
|
1072
|
)
|
|
1054
|
1073
|
|
|
@@
-1072,7
+1091,7
b' class CacheInvalidation(Base, BaseModel)'
|
|
1072
|
1091
|
__tablename__ = 'cache_invalidation'
|
|
1073
|
1092
|
__table_args__ = (
|
|
1074
|
1093
|
UniqueConstraint('cache_key'),
|
|
1075
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1094
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1076
|
1095
|
'mysql_charset': 'utf8'},
|
|
1077
|
1096
|
)
|
|
1078
|
1097
|
cache_id = Column("cache_id", Integer(), nullable=False, unique=True, default=None, primary_key=True)
|
|
@@
-1178,7
+1197,7
b' class CacheInvalidation(Base, BaseModel)'
|
|
1178
|
1197
|
class ChangesetComment(Base, BaseModel):
|
|
1179
|
1198
|
__tablename__ = 'changeset_comments'
|
|
1180
|
1199
|
__table_args__ = (
|
|
1181
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1200
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1182
|
1201
|
'mysql_charset': 'utf8'},
|
|
1183
|
1202
|
)
|
|
1184
|
1203
|
comment_id = Column('comment_id', Integer(), nullable=False, primary_key=True)
|
|
@@
-1207,10
+1226,48
b' class ChangesetComment(Base, BaseModel):'
|
|
1207
|
1226
|
.join(ChangesetComment.author).all()
|
|
1208
|
1227
|
|
|
1209
|
1228
|
|
|
|
1229
|
class ChangesetStatus(Base, BaseModel):
|
|
|
1230
|
__tablename__ = 'changeset_statuses'
|
|
|
1231
|
__table_args__ = (
|
|
|
1232
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
|
1233
|
'mysql_charset': 'utf8'}
|
|
|
1234
|
)
|
|
|
1235
|
|
|
|
1236
|
STATUSES = [
|
|
|
1237
|
('not_reviewed', _("Not Reviewed")), # (no icon) and default
|
|
|
1238
|
('approved', _("Approved")),
|
|
|
1239
|
('rejected', _("Rejected")),
|
|
|
1240
|
('under_review', _("Under Review")),
|
|
|
1241
|
]
|
|
|
1242
|
|
|
|
1243
|
changeset_status_id = Column('changeset_status_id', Integer(), nullable=False, primary_key=True)
|
|
|
1244
|
repo_id = Column('repo_id', Integer(), ForeignKey('repositories.repo_id'), nullable=False)
|
|
|
1245
|
user_id = Column("user_id", Integer(), ForeignKey('users.user_id'), nullable=False, unique=None)
|
|
|
1246
|
revision = Column('revision', String(40), nullable=False)
|
|
|
1247
|
status = Column('status', String(128), nullable=False, default=STATUSES[0][0])
|
|
|
1248
|
modified_at = Column('modified_at', DateTime(), nullable=False, default=datetime.datetime.now)
|
|
|
1249
|
|
|
|
1250
|
author = relationship('User', lazy='joined')
|
|
|
1251
|
repo = relationship('Repository')
|
|
|
1252
|
|
|
|
1253
|
|
|
|
1254
|
class ChangesetStatusHistory(Base, BaseModel):
|
|
|
1255
|
__tablename__ = 'changeset_statuses_history'
|
|
|
1256
|
__table_args__ = (
|
|
|
1257
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
|
1258
|
'mysql_charset': 'utf8'}
|
|
|
1259
|
)
|
|
|
1260
|
#TODO: check if sqla has a nice history table implementation
|
|
|
1261
|
changeset_status_id = Column('changeset_status_id', Integer(), ForeignKey('changeset_statuses.changeset_status_id'), nullable=False, primary_key=True)
|
|
|
1262
|
user_id = Column("user_id", Integer(), ForeignKey('users.user_id'), nullable=False, unique=None)
|
|
|
1263
|
status = Column('status', String(128), nullable=False)
|
|
|
1264
|
modified_at = Column('modified_at', DateTime(), nullable=False, default=datetime.datetime.now)
|
|
|
1265
|
|
|
|
1266
|
|
|
1210
|
1267
|
class Notification(Base, BaseModel):
|
|
1211
|
1268
|
__tablename__ = 'notifications'
|
|
1212
|
1269
|
__table_args__ = (
|
|
1213
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1270
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1214
|
1271
|
'mysql_charset': 'utf8'},
|
|
1215
|
1272
|
)
|
|
1216
|
1273
|
|
|
@@
-1264,7
+1321,7
b' class UserNotification(Base, BaseModel):'
|
|
1264
|
1321
|
__tablename__ = 'user_to_notification'
|
|
1265
|
1322
|
__table_args__ = (
|
|
1266
|
1323
|
UniqueConstraint('user_id', 'notification_id'),
|
|
1267
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1324
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1268
|
1325
|
'mysql_charset': 'utf8'}
|
|
1269
|
1326
|
)
|
|
1270
|
1327
|
user_id = Column('user_id', Integer(), ForeignKey('users.user_id'), primary_key=True)
|
|
@@
-1284,7
+1341,7
b' class UserNotification(Base, BaseModel):'
|
|
1284
|
1341
|
class DbMigrateVersion(Base, BaseModel):
|
|
1285
|
1342
|
__tablename__ = 'db_migrate_version'
|
|
1286
|
1343
|
__table_args__ = (
|
|
1287
|
|
{'extend_existing': True, 'mysql_engine':'InnoDB',
|
|
|
1344
|
{'extend_existing': True, 'mysql_engine': 'InnoDB',
|
|
1288
|
1345
|
'mysql_charset': 'utf8'},
|
|
1289
|
1346
|
)
|
|
1290
|
1347
|
repository_id = Column('repository_id', String(250), primary_key=True)
|