-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSQLTables.py
323 lines (287 loc) · 15.6 KB
/
SQLTables.py
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
# -*- coding: utf-8 -*-
# Copyright (c) 2006-2010 Stas Zykiewicz <[email protected]>
#
# SQLTables.py
# This program is free software; you can redistribute it and/or
# modify it under the terms of version 3 of the GNU General Public License
# as published by the Free Software Foundation. A copy of this license should
# be included in the file GPL-3.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Library General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
# Put all the tables that are used in the dbase here
# ACTIVITY DEVELOPERS, BE SURE YOU UNDERSTAND WHAT YOU MUST ADD HERE TO GET YOUR
# ACTIVITY UP AND RUNNING.
# PLEASE CONTACT THE AUTHOR OF THIS FILE OR THE SCHOOLSPLAY MAILINGLIST FOR
# ADDITIONAL INFO OR HELP.
# PLEASE DON'T CHANGE ANYTHING IN THIS MODULE WITHOUT DISCUSSING IT ON THE
# SCHOOLSPLAY MAILINGLIST.
# THE CreateTemplate SCRIPT and Datamanager DEPENDS ON CERTAIN SYNTAX TO BE
# PRESENT IN HERE.
# Make sure to add a ORM class to SPORMs.py if you add new sql tables.
import datetime
from sqlalchemy import *
import sqlalchemy.orm as sqlorm
import SPORMs as ORMS
def create_contentdb_orms(metadata):
tables = {}
for name in [k for k in ORMS.__dict__.keys() if k.startswith('game_')]:
t = Table(name, metadata, autoload=True)
orm = getattr(ORMS, name)
orm._name = name
sqlorm.mapper(orm, t)
tables[name] = orm
return tables
class SqlTables:
"""Class that holds the dbase table definitions and creates the tables for
the content dbase and user dbase."""
def _create_userdb_tables(self, metadata):
"""Called by the datamanager to setup the dbase tables.
We also map the ORM classes to the dbase tables.
Returns a dict with the keys set to the dbase table names and
the values set to the ORM classes."""
tables = {}
sqlorm.clear_mappers()
for table in self.tableslist:
self._appendcolumns(table)
if hasattr(ORMS, table.name):
orm = getattr(ORMS, table.name)
orm._name = table.name
sqlorm.mapper(orm, table)
tables[table.name] = orm
metadata.create_all()
return tables
def _appendcolumns(self, t):
# These are the skipped Tables for default columns
if t.name in ['users', 'options', 'activity_options', 'menu',
'served_content', 'group_names', 'change_pass', 'zorgenquete',
'spconf', 'users_faces', 'dt_sequence', 'stats', 'stats_session']:
return
# Columns added to all act tables
t.append_column(Column('table_id', Integer, primary_key=True))
t.append_column(Column('user_id', Integer, nullable=False))
t.append_column(Column('timespend', Unicode(6)))
t.append_column(Column('start_time', Unicode(17)))
t.append_column(Column('end_time', Unicode(17)))
t.append_column(Column('level', Integer))
t.append_column(Column('score', Float))
def __init__(self, metadata):
# table to hold user data, mandatory for any sp dbase.
# All other tables are regarded as activity tables.
# Used by the core, don't change it
self.users = Table('users', metadata,
Column('user_id', Integer, primary_key=True),
Column('login_name', Unicode(20), unique=True, nullable=False),
Column('title', Unicode(20)),
Column('first_name', Unicode(20)),
Column('last_name', Unicode(40)),
Column('birthdate', DateTime),
Column('group', Integer),
Column('profile', Unicode(200)),
Column('passwrd', Unicode(10)),
Column('activities', Unicode(250)),
Column('audio', Integer, default=50),
Column('dt_target', Unicode(250), default=u'default'),
Column('levelup_dlg', Unicode(4), default=u'true')
)
# The graph image needs two values:
# average (greek: mu-value) and the standard deviation (greek: sigma-value)
# mu is a integer between 1-10 and sigma is an integer between 0-100 which
# stand for the percentage deviation. (Probably always 50%)
self.activity_options = Table('activity_options', metadata,
Column('table_id', Integer, primary_key=True),
Column('activity', Unicode(20)),
Column('mu', Float, default=6.5),
Column('sigma', Float, default=1.63)
)
# served content table used by the acts to store content they serve so that
# we can make sure we don't serve the same content twice.
# This is meanly intended for the quizzes as the rest of the acts don't
# have much content, yet ?
self.served_content = Table('served_content', metadata,
Column('table_id', Integer, primary_key=True),
Column('user_id', Integer, nullable=False),
Column('CID', Integer, nullable=False),
Column('game_theme_id', Integer, nullable=False),
Column('module', Unicode(25)),
Column('start_time', DATETIME(timezone=True)),
Column('count_served', Integer, default=0),
Column('wrong', Integer, default=0)
)
# Table to convert group name to an id
self.group_names = Table('group_names', metadata,
Column('group_id', Integer, primary_key=True),
Column('group_name', Unicode(50), unique=True, nullable=False))
self.spconf = Table('spconf', metadata,
Column('table_id', Integer, primary_key=True),
Column('activity_name', Unicode(50)),
Column('key', Unicode(20)),
Column('value', Unicode(200)),
Column('theme', Unicode(20), default=u'default'),
Column('comment', Unicode(200)))
self.faces = Table('users_faces', metadata,
Column('face_id', Integer, primary_key=True),
Column('user_id', Integer),
Column('shape_diff', Float),
Column('temp_diff', Float),
Column('pic_name', Unicode(10)),
Column('datetime', DATETIME))
self.stats = Table('stats', metadata,
Column('ID', Integer, primary_key=True),
Column('datetime', DATETIME),
Column('activity_name', Unicode(50)),
Column('message', Unicode(255)),
Column('game_start_called', Boolean, default=False),
Column('game_nextlevel_called', Boolean, default=False),
Column('game_end_called', Boolean, default=False),
Column('is_cop', Boolean, default=False),
Column('user_id', Integer),
Column('final_score', Integer),
Column('session', Integer))
self.stats_session = Table('stats_session', metadata,
Column('ID', Integer, primary_key=True),
Column('datetime', DATETIME))
################ Activity tables ##########################
# Every activity *must* have a sql table set.
# The name of the table must be the same as the activity modules name
# The following cols are mandatory for any activity table.
# 'table_id' to hold the id
# 'user_id' to hold the user id (set by DM)
# 'timespend' to hold the time it took to complete the level. (set by core)
# 'start_time' to hold the time the level is started (set by core)
# 'end_time' to hold the time the level is finished (set by core)
# 'level' to hold the level number (set by core)
# 'score' to hold the score value.
# The mandatory columns are provided by the BaseTable class,
# the reporter tool expect them to be present so please extend the BaseTable
# instead of creating your own.
# The rest of the cols are set/used by the activity.
# LEAVE THESE COMMENTS IN HERE, THE CreateTemplate SCRIPT NEEDS THEM.
# @splitpoint@
self.math_test = Table('math_test', metadata)
self.flashcards = Table('flashcards', metadata)
self.BlockBreaker = Table('BlockBreaker', metadata)
self.TicTacToe = Table('TicTacToe', metadata)
self.fourrow = Table('fourrow', metadata,
Column('stonesplayed', Integer),
)
self.numbers_sp = Table('numbers_sp', metadata)
self.test_act = Table('test_act', metadata)
self.fishtank = Table('fishtank', metadata,
Column('fish', Integer),
Column('clearfish', Integer),
)
self.ichanger = Table('ichanger', metadata,
Column('wrongs', Integer),
Column('extra', Integer),
)
self.puzzle = Table('puzzle', metadata,
Column('total_pieces', Integer),
Column('wrongs', Integer),
)
self.simon_sp = Table('simon_sp', metadata,
Column('sounds', Integer))
self.soundmemory = Table('soundmemory', metadata,
Column('sounds', Integer),
Column('knownsounds', Integer),
)
self.memory_sp = Table('memory_sp', metadata,
Column('cards', Integer),
Column('knowncards', Integer),
)
self.quiz_royal = Table('quiz_royal', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz_history = Table('quiz_history', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz_picture = Table('quiz_picture', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz_math = Table('quiz_math', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz_text = Table('quiz_text', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz = Table('quiz', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz_general = Table('quiz_general', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz_melody = Table('quiz_melody', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.quiz_sayings = Table('quiz_sayings', metadata,
Column('total_questions', Integer),
Column('total_wrongs', Integer),
)
self.findit_sp = Table('findit_sp', metadata,
Column('total_diffs', Integer),
Column('wrongs', Integer),
)
self.electro_sp = Table('electro_sp', metadata,
Column('knowncards', Integer),
Column('cards', Integer))
self.video = Table('video', metadata,
Column('movie', Unicode(20)),
Column('answer', Integer))
self.dt_sequence = Table('dt_sequence', metadata,
Column('fortune', Integer),
Column('act_name', Unicode(255)),
Column('group', Unicode(255)),
Column('level', Integer),
Column('cycles', Integer),
Column('target', Unicode(255)),
Column('table_id', Integer, primary_key=True),
Column('order', Integer))
# TODO: This table is not finished, as soon as the specs are known we must update this
self.dltr = Table('dltr', metadata,
Column('activity', Unicode(20)),
Column('done', Integer),
Column('cycles', Integer),
Column('epoch', Numeric))
############ Setup an activity list #################################
# Now we put all the tables we have into this list.
# The Datamanager uses this list to setup the dbase tables.
self.tableslist = [self.users, self.activity_options, self.served_content,
self.spconf, self.stats, self.stats_session,
self.group_names, self.findit_sp, self.electro_sp,
self.quiz_text, self.quiz_melody, self.quiz_math,
self.quiz_picture, self.quiz_history, self.quiz_royal,
self.quiz, self.quiz_general, self.quiz_sayings,
self.memory_sp, self.simon_sp, self.puzzle,
self.soundmemory, self.ichanger, self.fishtank,
self.dltr,
self.test_act,
self.numbers_sp,
self.fourrow, self.dt_sequence,
self.TicTacToe,
self.BlockBreaker,
self.flashcards,
self.math_test]
if __name__ == '__main__':
# test code
from SPConstants import DBASEPATH
import sqlalchemy as sqla
engine = sqla.create_engine('sqlite:///%s' % DBASEPATH)
metadata = sqla.MetaData(engine)
# metadata.engine.echo = True
sqltb = SqlTables(metadata)
tables = sqltb._create_tables(metadata)
print(tables)