@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 CHANGED
@@ -1,5 +1,5 @@
1
1
  EXTENSION = pgpm-achievements
2
- DATA = sql/pgpm-achievements--0.15.3.sql
2
+ DATA = sql/pgpm-achievements--0.15.5.sql
3
3
 
4
4
  PG_CONFIG = pg_config
5
5
  PGXS := $(shell $(PG_CONFIG) --pgxs)
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@pgpm/achievements",
3
- "version": "0.21.2",
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.21.2",
25
- "@pgpm/verify": "0.21.2"
24
+ "@pgpm/jwt-claims": "0.23.0",
25
+ "@pgpm/verify": "0.23.0"
26
26
  },
27
27
  "devDependencies": {
28
- "pgpm": "^4.16.6"
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": "c7d836c99c7ce519e9bb79e6343bee3741781766"
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();