@pgpm/achievements 0.21.2 → 0.23.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/Makefile +1 -1
- package/package.json +5 -5
- package/sql/pgpm-achievements--0.15.5.sql +312 -0
package/Makefile
CHANGED
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pgpm/achievements",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.23.0",
|
|
4
4
|
"description": "Achievement system for tracking user progress and milestones",
|
|
5
5
|
"author": "Dan Lynch <pyramation@gmail.com>",
|
|
6
6
|
"contributors": [
|
|
@@ -21,11 +21,11 @@
|
|
|
21
21
|
"test:watch": "jest --watch"
|
|
22
22
|
},
|
|
23
23
|
"dependencies": {
|
|
24
|
-
"@pgpm/jwt-claims": "0.
|
|
25
|
-
"@pgpm/verify": "0.
|
|
24
|
+
"@pgpm/jwt-claims": "0.23.0",
|
|
25
|
+
"@pgpm/verify": "0.23.0"
|
|
26
26
|
},
|
|
27
27
|
"devDependencies": {
|
|
28
|
-
"pgpm": "^4.
|
|
28
|
+
"pgpm": "^4.23.2"
|
|
29
29
|
},
|
|
30
30
|
"repository": {
|
|
31
31
|
"type": "git",
|
|
@@ -35,5 +35,5 @@
|
|
|
35
35
|
"bugs": {
|
|
36
36
|
"url": "https://github.com/constructive-io/pgpm-modules/issues"
|
|
37
37
|
},
|
|
38
|
-
"gitHead": "
|
|
38
|
+
"gitHead": "c051ba5f98dd780c1f69e3ac07c2cc870d1c859d"
|
|
39
39
|
}
|
|
@@ -0,0 +1,312 @@
|
|
|
1
|
+
\echo Use "CREATE EXTENSION pgpm-achievements" to load this file. \quit
|
|
2
|
+
CREATE SCHEMA IF NOT EXISTS status_private;
|
|
3
|
+
|
|
4
|
+
GRANT USAGE ON SCHEMA status_private TO authenticated, anonymous;
|
|
5
|
+
|
|
6
|
+
ALTER DEFAULT PRIVILEGES IN SCHEMA status_private
|
|
7
|
+
GRANT EXECUTE ON FUNCTIONS TO authenticated;
|
|
8
|
+
|
|
9
|
+
CREATE SCHEMA IF NOT EXISTS status_public;
|
|
10
|
+
|
|
11
|
+
GRANT USAGE ON SCHEMA status_public TO authenticated, anonymous;
|
|
12
|
+
|
|
13
|
+
ALTER DEFAULT PRIVILEGES IN SCHEMA status_public
|
|
14
|
+
GRANT EXECUTE ON FUNCTIONS TO authenticated;
|
|
15
|
+
|
|
16
|
+
CREATE TABLE status_public.user_steps (
|
|
17
|
+
id uuid PRIMARY KEY DEFAULT uuidv7(),
|
|
18
|
+
user_id uuid NOT NULL,
|
|
19
|
+
name text NOT NULL,
|
|
20
|
+
count int NOT NULL DEFAULT 1,
|
|
21
|
+
created_at timestamptz NOT NULL DEFAULT CURRENT_TIMESTAMP
|
|
22
|
+
);
|
|
23
|
+
|
|
24
|
+
COMMENT ON TABLE status_public.user_steps IS 'The user achieving a requirement for a level. Log table that has every single step ever taken.';
|
|
25
|
+
|
|
26
|
+
COMMENT ON COLUMN status_public.user_steps.id IS 'Unique identifier for this step record';
|
|
27
|
+
|
|
28
|
+
COMMENT ON COLUMN status_public.user_steps.user_id IS 'User who performed this step';
|
|
29
|
+
|
|
30
|
+
COMMENT ON COLUMN status_public.user_steps.name IS 'Name of the level requirement this step counts toward';
|
|
31
|
+
|
|
32
|
+
COMMENT ON COLUMN status_public.user_steps.count IS 'Number of units this step contributes (default 1)';
|
|
33
|
+
|
|
34
|
+
COMMENT ON COLUMN status_public.user_steps.created_at IS 'Timestamp when this step was recorded';
|
|
35
|
+
|
|
36
|
+
CREATE INDEX ON status_public.user_steps (user_id, name);
|
|
37
|
+
|
|
38
|
+
CREATE FUNCTION status_private.user_completed_step(
|
|
39
|
+
step text,
|
|
40
|
+
user_id uuid DEFAULT jwt_public.current_user_id()
|
|
41
|
+
) RETURNS void AS $EOFCODE$
|
|
42
|
+
INSERT INTO status_public.user_steps ( name, user_id, count )
|
|
43
|
+
VALUES ( step, user_id, 1 );
|
|
44
|
+
$EOFCODE$ LANGUAGE sql VOLATILE SECURITY DEFINER;
|
|
45
|
+
|
|
46
|
+
CREATE FUNCTION status_private.user_incompleted_step(
|
|
47
|
+
step text,
|
|
48
|
+
user_id uuid DEFAULT jwt_public.current_user_id()
|
|
49
|
+
) RETURNS void AS $EOFCODE$
|
|
50
|
+
BEGIN
|
|
51
|
+
DELETE FROM status_public.user_steps s
|
|
52
|
+
WHERE s.user_id = user_incompleted_step.user_id
|
|
53
|
+
AND s.name = step;
|
|
54
|
+
DELETE FROM status_public.user_achievements a
|
|
55
|
+
WHERE a.user_id = user_incompleted_step.user_id
|
|
56
|
+
AND a.name = step;
|
|
57
|
+
END;
|
|
58
|
+
$EOFCODE$ LANGUAGE plpgsql VOLATILE SECURITY DEFINER;
|
|
59
|
+
|
|
60
|
+
CREATE FUNCTION status_private.tg_achievement() RETURNS trigger AS $EOFCODE$
|
|
61
|
+
DECLARE
|
|
62
|
+
is_null boolean;
|
|
63
|
+
task_name text;
|
|
64
|
+
BEGIN
|
|
65
|
+
IF (TG_OP = 'INSERT' OR TG_OP = 'UPDATE') THEN
|
|
66
|
+
task_name = TG_ARGV[1]::text;
|
|
67
|
+
EXECUTE format('SELECT ($1).%s IS NULL', TG_ARGV[0])
|
|
68
|
+
USING NEW INTO is_null;
|
|
69
|
+
IF (is_null IS FALSE) THEN
|
|
70
|
+
PERFORM status_private.user_completed_step(task_name);
|
|
71
|
+
END IF;
|
|
72
|
+
RETURN NEW;
|
|
73
|
+
END IF;
|
|
74
|
+
END;
|
|
75
|
+
$EOFCODE$ LANGUAGE plpgsql VOLATILE;
|
|
76
|
+
|
|
77
|
+
CREATE FUNCTION status_private.tg_achievement_toggle() RETURNS trigger AS $EOFCODE$
|
|
78
|
+
DECLARE
|
|
79
|
+
is_null boolean;
|
|
80
|
+
task_name text;
|
|
81
|
+
BEGIN
|
|
82
|
+
IF (TG_OP = 'INSERT' OR TG_OP = 'UPDATE') THEN
|
|
83
|
+
task_name = TG_ARGV[1]::text;
|
|
84
|
+
EXECUTE format('SELECT ($1).%s IS NULL', TG_ARGV[0])
|
|
85
|
+
USING NEW INTO is_null;
|
|
86
|
+
IF (is_null IS TRUE) THEN
|
|
87
|
+
PERFORM status_private.user_incompleted_step(task_name);
|
|
88
|
+
ELSE
|
|
89
|
+
PERFORM status_private.user_completed_step(task_name);
|
|
90
|
+
END IF;
|
|
91
|
+
RETURN NEW;
|
|
92
|
+
END IF;
|
|
93
|
+
END;
|
|
94
|
+
$EOFCODE$ LANGUAGE plpgsql VOLATILE;
|
|
95
|
+
|
|
96
|
+
CREATE FUNCTION status_private.tg_achievement_boolean() RETURNS trigger AS $EOFCODE$
|
|
97
|
+
DECLARE
|
|
98
|
+
is_true boolean;
|
|
99
|
+
task_name text;
|
|
100
|
+
BEGIN
|
|
101
|
+
IF (TG_OP = 'INSERT' OR TG_OP = 'UPDATE') THEN
|
|
102
|
+
task_name = TG_ARGV[1]::text;
|
|
103
|
+
EXECUTE format('SELECT ($1).%s IS TRUE', TG_ARGV[0])
|
|
104
|
+
USING NEW INTO is_true;
|
|
105
|
+
IF (is_true IS TRUE) THEN
|
|
106
|
+
PERFORM status_private.user_completed_step(task_name);
|
|
107
|
+
END IF;
|
|
108
|
+
RETURN NEW;
|
|
109
|
+
END IF;
|
|
110
|
+
END;
|
|
111
|
+
$EOFCODE$ LANGUAGE plpgsql VOLATILE;
|
|
112
|
+
|
|
113
|
+
CREATE FUNCTION status_private.tg_achievement_toggle_boolean() RETURNS trigger AS $EOFCODE$
|
|
114
|
+
DECLARE
|
|
115
|
+
is_true boolean;
|
|
116
|
+
task_name text;
|
|
117
|
+
BEGIN
|
|
118
|
+
IF (TG_OP = 'INSERT' OR TG_OP = 'UPDATE') THEN
|
|
119
|
+
task_name = TG_ARGV[1]::text;
|
|
120
|
+
EXECUTE format('SELECT ($1).%s IS TRUE', TG_ARGV[0])
|
|
121
|
+
USING NEW INTO is_true;
|
|
122
|
+
IF (is_true IS TRUE) THEN
|
|
123
|
+
PERFORM status_private.user_completed_step(task_name);
|
|
124
|
+
ELSE
|
|
125
|
+
PERFORM status_private.user_incompleted_step(task_name);
|
|
126
|
+
END IF;
|
|
127
|
+
RETURN NEW;
|
|
128
|
+
END IF;
|
|
129
|
+
END;
|
|
130
|
+
$EOFCODE$ LANGUAGE plpgsql VOLATILE;
|
|
131
|
+
|
|
132
|
+
CREATE TABLE status_public.user_achievements (
|
|
133
|
+
id uuid PRIMARY KEY DEFAULT uuidv7(),
|
|
134
|
+
user_id uuid NOT NULL,
|
|
135
|
+
name text NOT NULL,
|
|
136
|
+
count int NOT NULL DEFAULT 0,
|
|
137
|
+
created_at timestamptz NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
|
138
|
+
CONSTRAINT user_achievements_unique_key
|
|
139
|
+
UNIQUE (user_id, name)
|
|
140
|
+
);
|
|
141
|
+
|
|
142
|
+
COMMENT ON TABLE status_public.user_achievements IS 'This table represents the users progress for particular level requirements, tallying the total count. This table is updated via triggers and should not be updated maually.';
|
|
143
|
+
|
|
144
|
+
COMMENT ON COLUMN status_public.user_achievements.id IS 'Unique identifier for this achievement progress record';
|
|
145
|
+
|
|
146
|
+
COMMENT ON COLUMN status_public.user_achievements.user_id IS 'User whose progress is being tracked';
|
|
147
|
+
|
|
148
|
+
COMMENT ON COLUMN status_public.user_achievements.name IS 'Name of the level requirement this progress relates to';
|
|
149
|
+
|
|
150
|
+
COMMENT ON COLUMN status_public.user_achievements.count IS 'Accumulated count toward the requirement (updated by triggers)';
|
|
151
|
+
|
|
152
|
+
COMMENT ON COLUMN status_public.user_achievements.created_at IS 'Timestamp when this progress record was first created';
|
|
153
|
+
|
|
154
|
+
CREATE INDEX ON status_public.user_achievements (user_id, name);
|
|
155
|
+
|
|
156
|
+
CREATE FUNCTION status_private.upsert_achievement(
|
|
157
|
+
vuser_id uuid,
|
|
158
|
+
vname text,
|
|
159
|
+
vcount int
|
|
160
|
+
) RETURNS void AS $EOFCODE$
|
|
161
|
+
BEGIN
|
|
162
|
+
INSERT INTO status_public.user_achievements (user_id, name, count)
|
|
163
|
+
VALUES
|
|
164
|
+
(vuser_id, vname, GREATEST(vcount, 0))
|
|
165
|
+
ON CONFLICT ON CONSTRAINT user_achievements_unique_key
|
|
166
|
+
DO UPDATE SET
|
|
167
|
+
-- look ma! you can actually do aliases inside on conflict
|
|
168
|
+
count = user_achievements.count + EXCLUDED.count
|
|
169
|
+
;
|
|
170
|
+
END;
|
|
171
|
+
$EOFCODE$ LANGUAGE plpgsql VOLATILE;
|
|
172
|
+
|
|
173
|
+
CREATE TABLE status_public.levels (
|
|
174
|
+
name text NOT NULL PRIMARY KEY
|
|
175
|
+
);
|
|
176
|
+
|
|
177
|
+
COMMENT ON TABLE status_public.levels IS 'Levels for achievement';
|
|
178
|
+
|
|
179
|
+
COMMENT ON COLUMN status_public.levels.name IS 'Unique level name used as the primary key (e.g. bronze, silver, gold)';
|
|
180
|
+
|
|
181
|
+
GRANT SELECT ON status_public.levels TO PUBLIC;
|
|
182
|
+
|
|
183
|
+
CREATE TABLE status_public.level_requirements (
|
|
184
|
+
id uuid PRIMARY KEY DEFAULT uuidv7(),
|
|
185
|
+
name text NOT NULL,
|
|
186
|
+
level text NOT NULL,
|
|
187
|
+
required_count int DEFAULT 1,
|
|
188
|
+
priority int DEFAULT 100,
|
|
189
|
+
UNIQUE (name, level)
|
|
190
|
+
);
|
|
191
|
+
|
|
192
|
+
COMMENT ON TABLE status_public.level_requirements IS 'Requirements to achieve a level';
|
|
193
|
+
|
|
194
|
+
COMMENT ON COLUMN status_public.level_requirements.id IS 'Unique identifier for this requirement';
|
|
195
|
+
|
|
196
|
+
COMMENT ON COLUMN status_public.level_requirements.name IS 'Requirement name (e.g. posts_created, logins); matches user_steps.name';
|
|
197
|
+
|
|
198
|
+
COMMENT ON COLUMN status_public.level_requirements.level IS 'Level this requirement belongs to (references levels.name)';
|
|
199
|
+
|
|
200
|
+
COMMENT ON COLUMN status_public.level_requirements.required_count IS 'Number of steps needed to satisfy this requirement (default 1)';
|
|
201
|
+
|
|
202
|
+
COMMENT ON COLUMN status_public.level_requirements.priority IS 'Display/evaluation order; lower numbers are checked first (default 100)';
|
|
203
|
+
|
|
204
|
+
CREATE INDEX ON status_public.level_requirements (name, level, priority);
|
|
205
|
+
|
|
206
|
+
GRANT SELECT ON status_public.levels TO authenticated;
|
|
207
|
+
|
|
208
|
+
CREATE FUNCTION status_public.steps_required(
|
|
209
|
+
vlevel text,
|
|
210
|
+
vrole_id uuid DEFAULT jwt_public.current_user_id()
|
|
211
|
+
) RETURNS SETOF status_public.level_requirements AS $EOFCODE$
|
|
212
|
+
BEGIN
|
|
213
|
+
RETURN QUERY
|
|
214
|
+
SELECT
|
|
215
|
+
level_requirements.id,
|
|
216
|
+
level_requirements.name,
|
|
217
|
+
level_requirements.level,
|
|
218
|
+
-1*(coalesce(user_achievements.count,0)-level_requirements.required_count) as required_count,
|
|
219
|
+
level_requirements.priority
|
|
220
|
+
FROM
|
|
221
|
+
status_public.level_requirements
|
|
222
|
+
FULL OUTER JOIN status_public.user_achievements ON (
|
|
223
|
+
user_achievements.name = level_requirements.name
|
|
224
|
+
AND user_achievements.user_id =vrole_id
|
|
225
|
+
)
|
|
226
|
+
JOIN status_public.levels ON (level_requirements.level = levels.name)
|
|
227
|
+
WHERE
|
|
228
|
+
level_requirements.level = vlevel
|
|
229
|
+
AND -1*(coalesce(user_achievements.count,0)-level_requirements.required_count) > 0
|
|
230
|
+
ORDER BY priority ASC
|
|
231
|
+
;
|
|
232
|
+
END;
|
|
233
|
+
$EOFCODE$ LANGUAGE plpgsql STABLE;
|
|
234
|
+
|
|
235
|
+
CREATE FUNCTION status_public.user_achieved(
|
|
236
|
+
vlevel text,
|
|
237
|
+
vrole_id uuid DEFAULT jwt_public.current_user_id()
|
|
238
|
+
) RETURNS boolean AS $EOFCODE$
|
|
239
|
+
DECLARE
|
|
240
|
+
c int;
|
|
241
|
+
BEGIN
|
|
242
|
+
SELECT COUNT(*) FROM
|
|
243
|
+
status_public.steps_required(
|
|
244
|
+
vlevel,
|
|
245
|
+
vrole_id
|
|
246
|
+
)
|
|
247
|
+
INTO c;
|
|
248
|
+
|
|
249
|
+
RETURN c <= 0;
|
|
250
|
+
END;
|
|
251
|
+
$EOFCODE$ LANGUAGE plpgsql STABLE;
|
|
252
|
+
|
|
253
|
+
ALTER TABLE status_public.user_achievements
|
|
254
|
+
ENABLE ROW LEVEL SECURITY;
|
|
255
|
+
|
|
256
|
+
CREATE POLICY can_select_user_achievements
|
|
257
|
+
ON status_public.user_achievements
|
|
258
|
+
AS PERMISSIVE
|
|
259
|
+
FOR SELECT
|
|
260
|
+
TO PUBLIC
|
|
261
|
+
USING (
|
|
262
|
+
jwt_public.current_user_id() = user_id
|
|
263
|
+
);
|
|
264
|
+
|
|
265
|
+
CREATE POLICY can_insert_user_achievements
|
|
266
|
+
ON status_public.user_achievements
|
|
267
|
+
AS PERMISSIVE
|
|
268
|
+
FOR INSERT
|
|
269
|
+
TO PUBLIC
|
|
270
|
+
WITH CHECK (
|
|
271
|
+
false
|
|
272
|
+
);
|
|
273
|
+
|
|
274
|
+
CREATE POLICY can_update_user_achievements
|
|
275
|
+
ON status_public.user_achievements
|
|
276
|
+
AS PERMISSIVE
|
|
277
|
+
FOR UPDATE
|
|
278
|
+
TO PUBLIC
|
|
279
|
+
USING (
|
|
280
|
+
false
|
|
281
|
+
);
|
|
282
|
+
|
|
283
|
+
CREATE POLICY can_delete_user_achievements
|
|
284
|
+
ON status_public.user_achievements
|
|
285
|
+
AS PERMISSIVE
|
|
286
|
+
FOR DELETE
|
|
287
|
+
TO PUBLIC
|
|
288
|
+
USING (
|
|
289
|
+
false
|
|
290
|
+
);
|
|
291
|
+
|
|
292
|
+
GRANT INSERT ON status_public.user_achievements TO authenticated;
|
|
293
|
+
|
|
294
|
+
GRANT SELECT ON status_public.user_achievements TO authenticated;
|
|
295
|
+
|
|
296
|
+
GRANT UPDATE ON status_public.user_achievements TO authenticated;
|
|
297
|
+
|
|
298
|
+
GRANT DELETE ON status_public.user_achievements TO authenticated;
|
|
299
|
+
|
|
300
|
+
CREATE FUNCTION status_private.tg_update_achievements_tg() RETURNS trigger AS $EOFCODE$
|
|
301
|
+
DECLARE
|
|
302
|
+
BEGIN
|
|
303
|
+
PERFORM status_private.upsert_achievement(NEW.user_id, NEW.name, NEW.count);
|
|
304
|
+
RETURN NEW;
|
|
305
|
+
END;
|
|
306
|
+
$EOFCODE$ LANGUAGE plpgsql VOLATILE SECURITY DEFINER;
|
|
307
|
+
|
|
308
|
+
CREATE TRIGGER update_achievements_tg
|
|
309
|
+
AFTER INSERT
|
|
310
|
+
ON status_public.user_steps
|
|
311
|
+
FOR EACH ROW
|
|
312
|
+
EXECUTE PROCEDURE status_private.tg_update_achievements_tg();
|