@rimori/client 1.1.3 → 1.1.5

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/README.md CHANGED
@@ -67,6 +67,24 @@ export default {
67
67
  }
68
68
  ```
69
69
 
70
+ ### Vite Configuration
71
+
72
+ Add the following to your `vite.config.ts`:
73
+
74
+ ```javascript
75
+ import { defineConfig } from 'vite'
76
+ import react from '@vitejs/plugin-react'
77
+
78
+ export default defineConfig({
79
+ plugins: [react()],
80
+ base: './', // Set base path for proper asset loading
81
+ build: {
82
+ outDir: 'dist',
83
+ assetsDir: 'assets'
84
+ }
85
+ })
86
+ ```
87
+
70
88
  ## Core API - usePlugin Hook
71
89
 
72
90
  The `usePlugin()` hook is the main interface for accessing Rimori platform features:
@@ -558,6 +576,7 @@ community.sharedContent.getList<T>(contentType: string, filter?: SharedContentFi
558
576
  community.sharedContent.getNew<T>(contentType: string, instructions: SharedContentObjectRequest, filter?: SharedContentFilter, privateTopic?: boolean): Promise<BasicAssignment<T>>
559
577
  community.sharedContent.create<T>(content: SharedContent<T>): Promise<BasicAssignment<T>>
560
578
  community.sharedContent.update<T>(id: string, content: Partial<SharedContent<T>>): Promise<BasicAssignment<T>>
579
+ community.sharedContent.delete(id: string): Promise<BasicAssignment<any>>
561
580
  community.sharedContent.complete(contentType: string, assignmentId: string): Promise<void>
562
581
  ```
563
582
 
@@ -859,7 +878,7 @@ interface MyPluginData extends SharedContent<any> {
859
878
 
860
879
  The SharedContent has this type definition:
861
880
 
862
- ````
881
+ ```
863
882
  export interface SharedContent<T> {
864
883
  /** The type/category of the content (e.g. 'grammar_exercises', 'flashcards', etc.) */
865
884
  contentType: string;
@@ -996,4 +1015,3 @@ export default App;
996
1015
 
997
1016
  ## License
998
1017
 
999
- MIT License - see LICENSE file for details.
@@ -59,6 +59,13 @@ export declare class SharedContentController {
59
59
  * @throws {Error} if update fails.
60
60
  */
61
61
  updateSharedContent<T>(id: string, updates: Partial<SharedContent<T>>): Promise<BasicAssignment<T>>;
62
+ /**
63
+ * Soft delete shared content by setting the deleted_at timestamp.
64
+ * @param id - The ID of the content to delete.
65
+ * @returns The deleted shared content record.
66
+ * @throws {Error} if deletion fails or content not found.
67
+ */
68
+ deleteSharedContent(id: string): Promise<BasicAssignment<any>>;
62
69
  }
63
70
  /**
64
71
  * Interface representing shared content in the system.
@@ -28,6 +28,7 @@ export class SharedContentController {
28
28
  .select("*, scc:shared_content_completed(id)")
29
29
  .eq('content_type', contentType)
30
30
  .is('scc.id', null)
31
+ .is('deleted_at', null)
31
32
  .limit(10);
32
33
  if (filter) {
33
34
  query.contains('data', filter);
@@ -82,7 +83,8 @@ export class SharedContentController {
82
83
  const query = this.supabase.from("shared_content")
83
84
  .select("topic, keywords, scc:shared_content_completed(id)")
84
85
  .eq('content_type', contentType)
85
- .not('scc.id', 'is', null);
86
+ .not('scc.id', 'is', null)
87
+ .is('deleted_at', null);
86
88
  if (filter) {
87
89
  query.contains('data', filter);
88
90
  }
@@ -96,7 +98,7 @@ export class SharedContentController {
96
98
  }
97
99
  getSharedContent(contentType, id) {
98
100
  return __awaiter(this, void 0, void 0, function* () {
99
- const { data, error } = yield this.supabase.from("shared_content").select().eq('content_type', contentType).eq('id', id).single();
101
+ const { data, error } = yield this.supabase.from("shared_content").select().eq('content_type', contentType).eq('id', id).is('deleted_at', null).single();
100
102
  if (error) {
101
103
  console.error('error fetching shared content:', error);
102
104
  throw new Error('error fetching shared content');
@@ -118,7 +120,7 @@ export class SharedContentController {
118
120
  */
119
121
  getSharedContentList(contentType, filter, limit) {
120
122
  return __awaiter(this, void 0, void 0, function* () {
121
- const query = this.supabase.from("shared_content").select("*").eq('content_type', contentType).limit(limit !== null && limit !== void 0 ? limit : 30);
123
+ const query = this.supabase.from("shared_content").select("*").eq('content_type', contentType).is('deleted_at', null).limit(limit !== null && limit !== void 0 ? limit : 30);
122
124
  if (filter) {
123
125
  query.contains('data', filter);
124
126
  }
@@ -188,4 +190,27 @@ export class SharedContentController {
188
190
  return updatedContent[0];
189
191
  });
190
192
  }
193
+ /**
194
+ * Soft delete shared content by setting the deleted_at timestamp.
195
+ * @param id - The ID of the content to delete.
196
+ * @returns The deleted shared content record.
197
+ * @throws {Error} if deletion fails or content not found.
198
+ */
199
+ deleteSharedContent(id) {
200
+ return __awaiter(this, void 0, void 0, function* () {
201
+ const { data: deletedContent, error } = yield this.supabase
202
+ .from("shared_content")
203
+ .update({ deleted_at: new Date().toISOString() })
204
+ .eq('id', id)
205
+ .select();
206
+ if (error) {
207
+ console.error('error deleting shared content:', error);
208
+ throw new Error('error deleting shared content');
209
+ }
210
+ if (!deletedContent || deletedContent.length === 0) {
211
+ throw new Error('shared content not found or already deleted');
212
+ }
213
+ return deletedContent[0];
214
+ });
215
+ }
191
216
  }
@@ -172,6 +172,12 @@ export declare class RimoriClient {
172
172
  * @param assignmentId The id of the shared content item to complete.
173
173
  */
174
174
  complete: (contentType: string, assignmentId: string) => Promise<void>;
175
+ /**
176
+ * Delete a shared content item.
177
+ * @param id The id of the shared content item to delete.
178
+ * @returns The deleted shared content item.
179
+ */
180
+ delete: (id: string) => Promise<BasicAssignment<any>>;
175
181
  };
176
182
  };
177
183
  }
@@ -176,6 +176,14 @@ export class RimoriClient {
176
176
  */
177
177
  complete: (contentType, assignmentId) => __awaiter(this, void 0, void 0, function* () {
178
178
  return yield this.sharedContentController.completeSharedContent(contentType, assignmentId);
179
+ }),
180
+ /**
181
+ * Delete a shared content item.
182
+ * @param id The id of the shared content item to delete.
183
+ * @returns The deleted shared content item.
184
+ */
185
+ delete: (id) => __awaiter(this, void 0, void 0, function* () {
186
+ return yield this.sharedContentController.deleteSharedContent(id);
179
187
  })
180
188
  }
181
189
  };
@@ -17,7 +17,13 @@ export class StandaloneClient {
17
17
  static getInstance() {
18
18
  return __awaiter(this, void 0, void 0, function* () {
19
19
  if (!StandaloneClient.instance) {
20
- const config = yield fetch("http://localhost:3000/config.json").then(res => res.json());
20
+ const config = yield fetch("http://localhost:3000/config.json").then(res => res.json()).catch(err => {
21
+ console.error("Error fetching config.json", err);
22
+ return {
23
+ SUPABASE_URL: "https://pheptqdoqsdnadgoihvr.supabase.co",
24
+ SUPABASE_ANON_KEY: "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6InBoZXB0cWRvcXNkbmFkZ29paHZyIiwicm9sZSI6ImFub24iLCJpYXQiOjE3MzE2OTY2ODcsImV4cCI6MjA0NzI3MjY4N30.4GPFAXTF8685FaXISdAPNCIM-H3RGLo8GbyhQpu1mP0",
25
+ };
26
+ });
21
27
  StandaloneClient.instance = new StandaloneClient({ url: config.SUPABASE_URL, key: config.SUPABASE_ANON_KEY });
22
28
  }
23
29
  return StandaloneClient.instance;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@rimori/client",
3
- "version": "1.1.3",
3
+ "version": "1.1.5",
4
4
  "main": "dist/index.js",
5
5
  "types": "dist/index.d.ts",
6
6
  "exports": {
@@ -46,6 +46,7 @@ export class SharedContentController {
46
46
  .select("*, scc:shared_content_completed(id)")
47
47
  .eq('content_type', contentType)
48
48
  .is('scc.id', null)
49
+ .is('deleted_at', null)
49
50
  .limit(10);
50
51
 
51
52
  if (filter) {
@@ -113,6 +114,7 @@ export class SharedContentController {
113
114
  .select("topic, keywords, scc:shared_content_completed(id)")
114
115
  .eq('content_type', contentType)
115
116
  .not('scc.id', 'is', null)
117
+ .is('deleted_at', null)
116
118
 
117
119
  if (filter) {
118
120
  query.contains('data', filter);
@@ -128,7 +130,7 @@ export class SharedContentController {
128
130
  }
129
131
 
130
132
  public async getSharedContent<T>(contentType: string, id: string): Promise<BasicAssignment<T>> {
131
- const { data, error } = await this.supabase.from("shared_content").select().eq('content_type', contentType).eq('id', id).single();
133
+ const { data, error } = await this.supabase.from("shared_content").select().eq('content_type', contentType).eq('id', id).is('deleted_at', null).single();
132
134
  if (error) {
133
135
  console.error('error fetching shared content:', error);
134
136
  throw new Error('error fetching shared content');
@@ -148,7 +150,7 @@ export class SharedContentController {
148
150
  * @returns Array of shared content matching the criteria.
149
151
  */
150
152
  public async getSharedContentList<T>(contentType: string, filter?: SharedContentFilter, limit?: number): Promise<BasicAssignment<T>[]> {
151
- const query = this.supabase.from("shared_content").select("*").eq('content_type', contentType).limit(limit ?? 30);
153
+ const query = this.supabase.from("shared_content").select("*").eq('content_type', contentType).is('deleted_at', null).limit(limit ?? 30);
152
154
 
153
155
  if (filter) {
154
156
  query.contains('data', filter);
@@ -221,6 +223,31 @@ export class SharedContentController {
221
223
 
222
224
  return updatedContent[0];
223
225
  }
226
+
227
+ /**
228
+ * Soft delete shared content by setting the deleted_at timestamp.
229
+ * @param id - The ID of the content to delete.
230
+ * @returns The deleted shared content record.
231
+ * @throws {Error} if deletion fails or content not found.
232
+ */
233
+ public async deleteSharedContent(id: string): Promise<BasicAssignment<any>> {
234
+ const { data: deletedContent, error } = await this.supabase
235
+ .from("shared_content")
236
+ .update({ deleted_at: new Date().toISOString() })
237
+ .eq('id', id)
238
+ .select();
239
+
240
+ if (error) {
241
+ console.error('error deleting shared content:', error);
242
+ throw new Error('error deleting shared content');
243
+ }
244
+
245
+ if (!deletedContent || deletedContent.length === 0) {
246
+ throw new Error('shared content not found or already deleted');
247
+ }
248
+
249
+ return deletedContent[0];
250
+ }
224
251
  }
225
252
 
226
253
  /**
@@ -293,6 +293,14 @@ export class RimoriClient {
293
293
  */
294
294
  complete: async (contentType: string, assignmentId: string) => {
295
295
  return await this.sharedContentController.completeSharedContent(contentType, assignmentId);
296
+ },
297
+ /**
298
+ * Delete a shared content item.
299
+ * @param id The id of the shared content item to delete.
300
+ * @returns The deleted shared content item.
301
+ */
302
+ delete: async (id: string): Promise<BasicAssignment<any>> => {
303
+ return await this.sharedContentController.deleteSharedContent(id);
296
304
  }
297
305
  }
298
306
  }
@@ -18,7 +18,13 @@ export class StandaloneClient {
18
18
 
19
19
  public static async getInstance(): Promise<StandaloneClient> {
20
20
  if (!StandaloneClient.instance) {
21
- const config = await fetch("http://localhost:3000/config.json").then(res => res.json());
21
+ const config = await fetch("http://localhost:3000/config.json").then(res => res.json()).catch(err => {
22
+ console.error("Error fetching config.json", err);
23
+ return {
24
+ SUPABASE_URL: "https://pheptqdoqsdnadgoihvr.supabase.co",
25
+ SUPABASE_ANON_KEY: "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJzdXBhYmFzZSIsInJlZiI6InBoZXB0cWRvcXNkbmFkZ29paHZyIiwicm9sZSI6ImFub24iLCJpYXQiOjE3MzE2OTY2ODcsImV4cCI6MjA0NzI3MjY4N30.4GPFAXTF8685FaXISdAPNCIM-H3RGLo8GbyhQpu1mP0",
26
+ }
27
+ });
22
28
  StandaloneClient.instance = new StandaloneClient({ url: config.SUPABASE_URL, key: config.SUPABASE_ANON_KEY });
23
29
  }
24
30
  return StandaloneClient.instance;