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
|
/* Copyright 2016 Software Freedom Conservancy Inc.
*
* This software is licensed under the GNU Lesser General Public License
* (version 2.1 or later). See the COPYING file in this distribution.
*/
public struct TagID {
public const int64 INVALID = -1;
public int64 id;
public TagID(int64 id = INVALID) {
this.id = id;
}
public bool is_invalid() {
return (id == INVALID);
}
public bool is_valid() {
return (id != INVALID);
}
}
public class TagRow {
public TagID tag_id;
public string name;
public Gee.Set<string>? source_id_list;
public int64 time_created;
}
public class TagTable : DatabaseTable {
private static TagTable instance = null;
private TagTable() {
set_table_name("TagTable");
Sqlite.Statement stmt;
int res = db.prepare_v2("CREATE TABLE IF NOT EXISTS "
+ "TagTable "
+ "("
+ "id INTEGER PRIMARY KEY, "
+ "name TEXT UNIQUE NOT NULL, "
+ "photo_id_list TEXT, "
+ "time_created INTEGER"
+ ")", -1, out stmt);
assert(res == Sqlite.OK);
res = stmt.step();
if (res != Sqlite.DONE)
fatal("create TagTable", res);
}
public static TagTable get_instance() {
if (instance == null)
instance = new TagTable();
return instance;
}
public static void upgrade_for_htags() {
TagTable table = get_instance();
try {
Gee.List<TagRow?> rows = table.get_all_rows();
foreach (TagRow row in rows) {
row.name = row.name.replace(Tag.PATH_SEPARATOR_STRING, "-");
table.rename(row.tag_id, row.name);
}
} catch (DatabaseError e) {
error ("TagTable: can't upgrade tag names for hierarchical tag support: %s", e.message);
}
}
public TagRow add(string name) throws DatabaseError {
Sqlite.Statement stmt;
int res = db.prepare_v2("INSERT INTO TagTable (name, time_created) VALUES (?, ?)", -1,
out stmt);
assert(res == Sqlite.OK);
var time_created = now_sec();
res = stmt.bind_text(1, name);
assert(res == Sqlite.OK);
res = stmt.bind_int64(2, time_created);
assert(res == Sqlite.OK);
res = stmt.step();
if (res != Sqlite.DONE)
throw_error("TagTable.add", res);
TagRow row = new TagRow();
row.tag_id = TagID(db.last_insert_rowid());
row.name = name;
row.source_id_list = null;
row.time_created = time_created;
return row;
}
// All fields but tag_id are respected in TagRow.
public TagID create_from_row(TagRow row) throws DatabaseError {
Sqlite.Statement stmt;
int res = db.prepare_v2("INSERT INTO TagTable (name, photo_id_list, time_created) VALUES (?, ?, ?)",
-1, out stmt);
assert(res == Sqlite.OK);
res = stmt.bind_text(1, row.name);
assert(res == Sqlite.OK);
res = stmt.bind_text(2, serialize_source_ids(row.source_id_list));
assert(res == Sqlite.OK);
res = stmt.bind_int64(3, row.time_created);
assert(res == Sqlite.OK);
res = stmt.step();
if (res != Sqlite.DONE)
throw_error("TagTable.create_from_row", res);
return TagID(db.last_insert_rowid());
}
public void remove(TagID tag_id) throws DatabaseError {
delete_by_id(tag_id.id);
}
public string? get_name(TagID tag_id) throws DatabaseError {
Sqlite.Statement stmt;
if (!select_by_id(tag_id.id, "name", out stmt))
return null;
return stmt.column_text(0);
}
public TagRow? get_row(TagID tag_id) throws DatabaseError {
Sqlite.Statement stmt;
int res = db.prepare_v2("SELECT name, photo_id_list, time_created FROM TagTable WHERE id=?",
-1, out stmt);
assert(res == Sqlite.OK);
res = stmt.bind_int64(1, tag_id.id);
assert(res == Sqlite.OK);
res = stmt.step();
if (res == Sqlite.DONE)
return null;
else if (res != Sqlite.ROW)
throw_error("TagTable.get_row", res);
TagRow row = new TagRow();
row.tag_id = tag_id;
row.name = stmt.column_text(0);
row.source_id_list = unserialize_source_ids(stmt.column_text(1));
row.time_created = stmt.column_int64(2);
return row;
}
public Gee.List<TagRow?> get_all_rows() throws DatabaseError {
Sqlite.Statement stmt;
int res = db.prepare_v2("SELECT id, name, photo_id_list, time_created FROM TagTable", -1,
out stmt);
assert(res == Sqlite.OK);
Gee.List<TagRow?> rows = new Gee.ArrayList<TagRow?>();
for (;;) {
res = stmt.step();
if (res == Sqlite.DONE)
break;
else if (res != Sqlite.ROW)
throw_error("TagTable.get_all_rows", res);
// res == Sqlite.ROW
TagRow row = new TagRow();
row.tag_id = TagID(stmt.column_int64(0));
row.name = stmt.column_text(1);
row.source_id_list = unserialize_source_ids(stmt.column_text(2));
row.time_created = stmt.column_int64(3);
rows.add(row);
}
return rows;
}
public void rename(TagID tag_id, string new_name) throws DatabaseError {
update_text_by_id_2(tag_id.id, "name", new_name);
}
public void set_tagged_sources(TagID tag_id, Gee.Collection<string> source_ids) throws DatabaseError {
Sqlite.Statement stmt;
int res = db.prepare_v2("UPDATE TagTable SET photo_id_list=? WHERE id=?", -1, out stmt);
assert(res == Sqlite.OK);
res = stmt.bind_text(1, serialize_source_ids(source_ids));
assert(res == Sqlite.OK);
res = stmt.bind_int64(2, tag_id.id);
assert(res == Sqlite.OK);
res = stmt.step();
if (res != Sqlite.DONE)
throw_error("TagTable.set_tagged_photos", res);
}
private string? serialize_source_ids(Gee.Collection<string>? source_ids) {
if (source_ids == null)
return null;
StringBuilder result = new StringBuilder();
foreach (string source_id in source_ids) {
result.append(source_id);
result.append(",");
}
return (result.len != 0) ? result.str : null;
}
private Gee.Set<string> unserialize_source_ids(string? text_list) {
Gee.Set<string> result = new Gee.HashSet<string>();
if (text_list == null)
return result;
string[] split = text_list.split(",");
foreach (string token in split) {
if (is_string_empty(token))
continue;
// handle current and legacy encoding of source ids -- in the past, we only stored
// LibraryPhotos in tags so we only needed to store the numeric database key of the
// photo to uniquely identify it. Now, however, tags can store arbitrary MediaSources,
// so instead of simply storing a number we store the source id, a string that contains
// a typename followed by an identifying number (e.g., "video-022354").
if (token[0].isdigit()) {
// this is a legacy entry
result.add(PhotoID.upgrade_photo_id_to_source_id(PhotoID(int64.parse(token, 10))));
} else if (token[0].isalpha()) {
// this is a modern entry
result.add(token);
}
}
return result;
}
}
|