@trafficgroup/knex-rel 0.1.11 → 0.1.12

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.
Files changed (110) hide show
  1. package/.claude/settings.local.json +2 -5
  2. package/CLAUDE.md +11 -2
  3. package/dist/constants/folder.constants.d.ts +12 -0
  4. package/dist/constants/folder.constants.js +28 -0
  5. package/dist/constants/folder.constants.js.map +1 -0
  6. package/dist/constants/video.constants.d.ts +2 -2
  7. package/dist/constants/video.constants.js +9 -5
  8. package/dist/constants/video.constants.js.map +1 -1
  9. package/dist/dao/VideoMinuteResultDAO.d.ts +1 -1
  10. package/dist/dao/VideoMinuteResultDAO.js +29 -23
  11. package/dist/dao/VideoMinuteResultDAO.js.map +1 -1
  12. package/dist/dao/auth/auth.dao.js +4 -1
  13. package/dist/dao/auth/auth.dao.js.map +1 -1
  14. package/dist/dao/batch/batch.dao.js +13 -14
  15. package/dist/dao/batch/batch.dao.js.map +1 -1
  16. package/dist/dao/camera/camera.dao.js +10 -7
  17. package/dist/dao/camera/camera.dao.js.map +1 -1
  18. package/dist/dao/chat/chat.dao.d.ts +1 -1
  19. package/dist/dao/chat/chat.dao.js +27 -40
  20. package/dist/dao/chat/chat.dao.js.map +1 -1
  21. package/dist/dao/folder/folder.dao.d.ts +10 -1
  22. package/dist/dao/folder/folder.dao.js +44 -6
  23. package/dist/dao/folder/folder.dao.js.map +1 -1
  24. package/dist/dao/location/location.dao.js +16 -9
  25. package/dist/dao/location/location.dao.js.map +1 -1
  26. package/dist/dao/message/message.dao.d.ts +1 -1
  27. package/dist/dao/message/message.dao.js +18 -26
  28. package/dist/dao/message/message.dao.js.map +1 -1
  29. package/dist/dao/report-configuration/report-configuration.dao.js +32 -31
  30. package/dist/dao/report-configuration/report-configuration.dao.js.map +1 -1
  31. package/dist/dao/study/study.dao.js +7 -2
  32. package/dist/dao/study/study.dao.js.map +1 -1
  33. package/dist/dao/user/user.dao.js +4 -1
  34. package/dist/dao/user/user.dao.js.map +1 -1
  35. package/dist/dao/user-push-notification-token/user-push-notification-token.dao.js +26 -8
  36. package/dist/dao/user-push-notification-token/user-push-notification-token.dao.js.map +1 -1
  37. package/dist/dao/video/video.dao.js +30 -28
  38. package/dist/dao/video/video.dao.js.map +1 -1
  39. package/dist/index.d.ts +8 -5
  40. package/dist/index.js +5 -1
  41. package/dist/index.js.map +1 -1
  42. package/dist/interfaces/batch/batch.interfaces.d.ts +1 -1
  43. package/dist/interfaces/camera/camera.interfaces.d.ts +1 -1
  44. package/dist/interfaces/chat/chat.interfaces.d.ts +3 -3
  45. package/dist/interfaces/folder/folder.interfaces.d.ts +1 -1
  46. package/dist/interfaces/message/message.interfaces.d.ts +2 -2
  47. package/dist/interfaces/study/study.interfaces.d.ts +2 -2
  48. package/dist/interfaces/user/user.interfaces.d.ts +1 -1
  49. package/dist/interfaces/user-push-notification-token/user-push-notification-token.interfaces.d.ts +1 -1
  50. package/dist/interfaces/video/video.interfaces.d.ts +2 -2
  51. package/migrations/20250717160737_migration.ts +1 -1
  52. package/migrations/20250717160908_migration.ts +5 -2
  53. package/migrations/20250717161310_migration.ts +1 -1
  54. package/migrations/20250717161406_migration.ts +3 -3
  55. package/migrations/20250717162431_migration.ts +1 -1
  56. package/migrations/20250717173228_migration.ts +2 -2
  57. package/migrations/20250717204731_migration.ts +1 -1
  58. package/migrations/20250722210109_migration.ts +8 -4
  59. package/migrations/20250722211019_migration.ts +1 -1
  60. package/migrations/20250723153852_migration.ts +13 -10
  61. package/migrations/20250723162257_migration.ts +4 -7
  62. package/migrations/20250723171109_migration.ts +4 -7
  63. package/migrations/20250723205331_migration.ts +6 -9
  64. package/migrations/20250724191345_migration.ts +8 -11
  65. package/migrations/20250730180932_migration.ts +14 -13
  66. package/migrations/20250730213625_migration.ts +8 -11
  67. package/migrations/20250804124509_migration.ts +26 -21
  68. package/migrations/20250804132053_migration.ts +5 -8
  69. package/migrations/20250804164518_migration.ts +7 -7
  70. package/migrations/20250823223016_migration.ts +32 -21
  71. package/migrations/20250910015452_migration.ts +18 -6
  72. package/migrations/20250911000000_migration.ts +18 -4
  73. package/migrations/20250917144153_migration.ts +14 -7
  74. package/migrations/20250930200521_migration.ts +8 -4
  75. package/migrations/20251010143500_migration.ts +27 -6
  76. package/migrations/20251020225758_migration.ts +51 -15
  77. package/migrations/20251112120000_migration.ts +10 -2
  78. package/migrations/20251112120200_migration.ts +19 -7
  79. package/migrations/20251112120300_migration.ts +7 -2
  80. package/package.json +1 -1
  81. package/src/constants/folder.constants.ts +29 -0
  82. package/src/constants/video.constants.ts +11 -7
  83. package/src/d.types.ts +18 -14
  84. package/src/dao/VideoMinuteResultDAO.ts +72 -49
  85. package/src/dao/auth/auth.dao.ts +58 -55
  86. package/src/dao/batch/batch.dao.ts +101 -98
  87. package/src/dao/camera/camera.dao.ts +124 -121
  88. package/src/dao/chat/chat.dao.ts +45 -45
  89. package/src/dao/folder/folder.dao.ts +112 -55
  90. package/src/dao/location/location.dao.ts +109 -87
  91. package/src/dao/message/message.dao.ts +32 -32
  92. package/src/dao/report-configuration/report-configuration.dao.ts +370 -342
  93. package/src/dao/study/study.dao.ts +88 -63
  94. package/src/dao/user/user.dao.ts +52 -50
  95. package/src/dao/user-push-notification-token/user-push-notification-token.dao.ts +80 -48
  96. package/src/dao/video/video.dao.ts +385 -334
  97. package/src/entities/BaseEntity.ts +1 -1
  98. package/src/index.ts +42 -17
  99. package/src/interfaces/auth/auth.interfaces.ts +10 -10
  100. package/src/interfaces/batch/batch.interfaces.ts +1 -1
  101. package/src/interfaces/camera/camera.interfaces.ts +9 -9
  102. package/src/interfaces/chat/chat.interfaces.ts +4 -4
  103. package/src/interfaces/folder/folder.interfaces.ts +2 -2
  104. package/src/interfaces/location/location.interfaces.ts +7 -7
  105. package/src/interfaces/message/message.interfaces.ts +3 -3
  106. package/src/interfaces/report-configuration/report-configuration.interfaces.ts +16 -16
  107. package/src/interfaces/study/study.interfaces.ts +3 -3
  108. package/src/interfaces/user/user.interfaces.ts +9 -9
  109. package/src/interfaces/user-push-notification-token/user-push-notification-token.interfaces.ts +9 -9
  110. package/src/interfaces/video/video.interfaces.ts +34 -34
@@ -4,131 +4,134 @@ import { ICamera } from "../../interfaces/camera/camera.interfaces";
4
4
  import KnexManager from "../../KnexConnection";
5
5
 
6
6
  export class CameraDAO implements IBaseDAO<ICamera> {
7
- private _knex: Knex<any, unknown[]> = KnexManager.getConnection();
8
-
9
- async create(item: ICamera): Promise<ICamera> {
10
- const [createdCamera] = await this._knex("cameras").insert(item).returning("*");
11
- return createdCamera;
12
- }
13
-
14
- async getById(id: number): Promise<ICamera | null> {
15
- const camera = await this._knex("cameras").where({ id }).first();
16
- return camera || null;
17
- }
18
-
19
- async getByUuid(uuid: string): Promise<ICamera | null> {
20
- const camera = await this._knex("cameras").where({ uuid }).first();
21
- return camera || null;
22
- }
23
-
24
- async update(id: number, item: Partial<ICamera>): Promise<ICamera | null> {
25
- const [updatedCamera] = await this._knex("cameras").where({ id }).update(item).returning("*");
26
- return updatedCamera || null;
27
- }
28
-
29
- async delete(id: number): Promise<boolean> {
30
- const result = await this._knex("cameras").where({ id }).del();
31
- return result > 0;
7
+ private _knex: Knex<any, unknown[]> = KnexManager.getConnection();
8
+
9
+ async create(item: ICamera): Promise<ICamera> {
10
+ const [createdCamera] = await this._knex("cameras")
11
+ .insert(item)
12
+ .returning("*");
13
+ return createdCamera;
14
+ }
15
+
16
+ async getById(id: number): Promise<ICamera | null> {
17
+ const camera = await this._knex("cameras").where({ id }).first();
18
+ return camera || null;
19
+ }
20
+
21
+ async getByUuid(uuid: string): Promise<ICamera | null> {
22
+ const camera = await this._knex("cameras").where({ uuid }).first();
23
+ return camera || null;
24
+ }
25
+
26
+ async update(id: number, item: Partial<ICamera>): Promise<ICamera | null> {
27
+ const [updatedCamera] = await this._knex("cameras")
28
+ .where({ id })
29
+ .update(item)
30
+ .returning("*");
31
+ return updatedCamera || null;
32
+ }
33
+
34
+ async delete(id: number): Promise<boolean> {
35
+ const result = await this._knex("cameras").where({ id }).del();
36
+ return result > 0;
37
+ }
38
+
39
+ async getAll(page: number, limit: number): Promise<IDataPaginator<ICamera>> {
40
+ const offset = (page - 1) * limit;
41
+
42
+ const [countResult] = await this._knex("cameras").count("* as count");
43
+ const totalCount = +countResult.count;
44
+ const cameras = await this._knex("cameras").limit(limit).offset(offset);
45
+
46
+ return {
47
+ success: true,
48
+ data: cameras,
49
+ page,
50
+ limit,
51
+ count: cameras.length,
52
+ totalCount,
53
+ totalPages: Math.ceil(totalCount / limit),
54
+ };
55
+ }
56
+
57
+ /**
58
+ * Get cameras for a specific location
59
+ * @param locationId - The location ID
60
+ * @param status - Optional filter by camera status
61
+ */
62
+ async getCamerasByLocationId(
63
+ locationId: number,
64
+ status?: string,
65
+ ): Promise<ICamera[]> {
66
+ let query = this._knex("cameras").where("locationId", locationId);
67
+
68
+ if (status) {
69
+ query = query.where("status", status);
32
70
  }
33
71
 
34
- async getAll(page: number, limit: number): Promise<IDataPaginator<ICamera>> {
35
- const offset = (page - 1) * limit;
36
-
37
- const [countResult] = await this._knex("cameras").count("* as count");
38
- const totalCount = +countResult.count;
39
- const cameras = await this._knex("cameras").limit(limit).offset(offset);
40
-
41
- return {
42
- success: true,
43
- data: cameras,
44
- page,
45
- limit,
46
- count: cameras.length,
47
- totalCount,
48
- totalPages: Math.ceil(totalCount / limit),
49
- };
72
+ return query.orderBy("name", "asc");
73
+ }
74
+
75
+ /**
76
+ * Get cameras for a specific location with pagination
77
+ * @param locationId - The location ID
78
+ * @param page - Page number
79
+ * @param limit - Items per page
80
+ * @param status - Optional filter by camera status
81
+ */
82
+ async getCamerasByLocationIdPaginated(
83
+ locationId: number,
84
+ page: number,
85
+ limit: number,
86
+ status?: string,
87
+ ): Promise<IDataPaginator<ICamera>> {
88
+ const offset = (page - 1) * limit;
89
+
90
+ let query = this._knex("cameras").where("locationId", locationId);
91
+
92
+ if (status) {
93
+ query = query.where("status", status);
50
94
  }
51
95
 
52
- /**
53
- * Get cameras for a specific location
54
- * @param locationId - The location ID
55
- * @param status - Optional filter by camera status
56
- */
57
- async getCamerasByLocationId(
58
- locationId: number,
59
- status?: string
60
- ): Promise<ICamera[]> {
61
- let query = this._knex("cameras")
62
- .where("locationId", locationId);
63
-
64
- if (status) {
65
- query = query.where("status", status);
66
- }
67
-
68
- return query.orderBy("name", "asc");
96
+ const [countResult] = await query.clone().count("* as count");
97
+ const totalCount = +countResult.count;
98
+ const cameras = await query
99
+ .clone()
100
+ .limit(limit)
101
+ .offset(offset)
102
+ .orderBy("name", "asc");
103
+
104
+ return {
105
+ success: true,
106
+ data: cameras,
107
+ page,
108
+ limit,
109
+ count: cameras.length,
110
+ totalCount,
111
+ totalPages: Math.ceil(totalCount / limit),
112
+ };
113
+ }
114
+
115
+ /**
116
+ * Check if a camera name already exists for a location
117
+ * @param name - Camera name
118
+ * @param locationId - Location ID
119
+ * @param excludeId - Optional camera ID to exclude (for update operations)
120
+ */
121
+ async existsByNameAndLocation(
122
+ name: string,
123
+ locationId: number,
124
+ excludeId?: number,
125
+ ): Promise<boolean> {
126
+ let query = this._knex("cameras")
127
+ .where("locationId", locationId)
128
+ .where("name", name);
129
+
130
+ if (excludeId !== undefined) {
131
+ query = query.whereNot("id", excludeId);
69
132
  }
70
133
 
71
- /**
72
- * Get cameras for a specific location with pagination
73
- * @param locationId - The location ID
74
- * @param page - Page number
75
- * @param limit - Items per page
76
- * @param status - Optional filter by camera status
77
- */
78
- async getCamerasByLocationIdPaginated(
79
- locationId: number,
80
- page: number,
81
- limit: number,
82
- status?: string
83
- ): Promise<IDataPaginator<ICamera>> {
84
- const offset = (page - 1) * limit;
85
-
86
- let query = this._knex("cameras")
87
- .where("locationId", locationId);
88
-
89
- if (status) {
90
- query = query.where("status", status);
91
- }
92
-
93
- const [countResult] = await query.clone().count("* as count");
94
- const totalCount = +countResult.count;
95
- const cameras = await query
96
- .clone()
97
- .limit(limit)
98
- .offset(offset)
99
- .orderBy("name", "asc");
100
-
101
- return {
102
- success: true,
103
- data: cameras,
104
- page,
105
- limit,
106
- count: cameras.length,
107
- totalCount,
108
- totalPages: Math.ceil(totalCount / limit)
109
- };
110
- }
111
-
112
- /**
113
- * Check if a camera name already exists for a location
114
- * @param name - Camera name
115
- * @param locationId - Location ID
116
- * @param excludeId - Optional camera ID to exclude (for update operations)
117
- */
118
- async existsByNameAndLocation(
119
- name: string,
120
- locationId: number,
121
- excludeId?: number
122
- ): Promise<boolean> {
123
- let query = this._knex("cameras")
124
- .where("locationId", locationId)
125
- .where("name", name);
126
-
127
- if (excludeId !== undefined) {
128
- query = query.whereNot("id", excludeId);
129
- }
130
-
131
- const result = await query.first();
132
- return result !== null && result !== undefined;
133
- }
134
+ const result = await query.first();
135
+ return result !== null && result !== undefined;
136
+ }
134
137
  }
@@ -1,6 +1,10 @@
1
1
  import { Knex } from "knex";
2
2
  import { IBaseDAO, IDataPaginator } from "../../d.types";
3
- import { IChat, IChatCreate, IChatUpdate } from '../../interfaces/chat/chat.interfaces';
3
+ import {
4
+ IChat,
5
+ IChatCreate,
6
+ IChatUpdate,
7
+ } from "../../interfaces/chat/chat.interfaces";
4
8
  import KnexManager from "../../KnexConnection";
5
9
 
6
10
  export interface IChatPaginatorResponse extends IDataPaginator<IChat> {
@@ -11,23 +15,17 @@ export class ChatDAO implements IBaseDAO<IChat> {
11
15
  private _knex: Knex<any, unknown[]> = KnexManager.getConnection();
12
16
 
13
17
  async create(item: IChatCreate): Promise<IChat> {
14
- const [result] = await this._knex('chat')
15
- .insert(item)
16
- .returning('*');
18
+ const [result] = await this._knex("chat").insert(item).returning("*");
17
19
  return result;
18
20
  }
19
21
 
20
22
  async getById(id: number): Promise<IChat | null> {
21
- const result = await this._knex('chat')
22
- .where('id', id)
23
- .first();
23
+ const result = await this._knex("chat").where("id", id).first();
24
24
  return result || null;
25
25
  }
26
26
 
27
27
  async getByUuid(uuid: string): Promise<IChat | null> {
28
- const result = await this._knex('chat')
29
- .where('uuid', uuid)
30
- .first();
28
+ const result = await this._knex("chat").where("uuid", uuid).first();
31
29
  return result || null;
32
30
  }
33
31
 
@@ -35,11 +33,11 @@ export class ChatDAO implements IBaseDAO<IChat> {
35
33
  const offset = (page - 1) * limit;
36
34
 
37
35
  const [results, [{ count }]] = await Promise.all([
38
- this._knex('chat')
39
- .orderBy('created_at', 'desc')
36
+ this._knex("chat")
37
+ .orderBy("created_at", "desc")
40
38
  .limit(limit)
41
39
  .offset(offset),
42
- this._knex('chat').count('* as count')
40
+ this._knex("chat").count("* as count"),
43
41
  ]);
44
42
 
45
43
  const totalCount = parseInt(count as string);
@@ -50,40 +48,40 @@ export class ChatDAO implements IBaseDAO<IChat> {
50
48
  limit,
51
49
  count: results.length,
52
50
  totalCount,
53
- totalPages: Math.ceil(totalCount / limit)
51
+ totalPages: Math.ceil(totalCount / limit),
54
52
  };
55
53
  }
56
54
 
57
55
  async update(id: number, item: IChatUpdate): Promise<IChat | null> {
58
- const [result] = await this._knex('chat')
59
- .where('id', id)
56
+ const [result] = await this._knex("chat")
57
+ .where("id", id)
60
58
  .update({
61
59
  ...item,
62
- updated_at: new Date()
60
+ updated_at: new Date(),
63
61
  })
64
- .returning('*');
62
+ .returning("*");
65
63
  return result || null;
66
64
  }
67
65
 
68
66
  async delete(id: number): Promise<boolean> {
69
- const result = await this._knex('chat')
70
- .where('id', id)
71
- .delete();
67
+ const result = await this._knex("chat").where("id", id).delete();
72
68
  return result > 0;
73
69
  }
74
70
 
75
- async getByUserId(userId: number, page = 1, limit = 10): Promise<IDataPaginator<IChat>> {
71
+ async getByUserId(
72
+ userId: number,
73
+ page = 1,
74
+ limit = 10,
75
+ ): Promise<IDataPaginator<IChat>> {
76
76
  const offset = (page - 1) * limit;
77
77
 
78
78
  const [results, [{ count }]] = await Promise.all([
79
- this._knex('chat')
80
- .where('userId', userId)
81
- .orderBy('created_at', 'desc')
79
+ this._knex("chat")
80
+ .where("userId", userId)
81
+ .orderBy("created_at", "desc")
82
82
  .limit(limit)
83
83
  .offset(offset),
84
- this._knex('chat')
85
- .where('userId', userId)
86
- .count('* as count')
84
+ this._knex("chat").where("userId", userId).count("* as count"),
87
85
  ]);
88
86
 
89
87
  const totalCount = parseInt(count as string);
@@ -94,36 +92,38 @@ export class ChatDAO implements IBaseDAO<IChat> {
94
92
  limit,
95
93
  count: results.length,
96
94
  totalCount,
97
- totalPages: Math.ceil(totalCount / limit)
95
+ totalPages: Math.ceil(totalCount / limit),
98
96
  };
99
97
  }
100
98
 
101
- async getActiveByUserId(userId: number, page = 1, limit = 10, query?: string): Promise<IChatPaginatorResponse> {
99
+ async getActiveByUserId(
100
+ userId: number,
101
+ page = 1,
102
+ limit = 10,
103
+ query?: string,
104
+ ): Promise<IChatPaginatorResponse> {
102
105
  const offset = (page - 1) * limit;
103
106
 
104
107
  // Build data query
105
- let dataQuery = this._knex('chat')
106
- .where('userId', userId)
107
- .where('status', 'active');
108
+ let dataQuery = this._knex("chat")
109
+ .where("userId", userId)
110
+ .where("status", "active");
108
111
 
109
112
  // Build count query
110
- let countQuery = this._knex('chat')
111
- .where('userId', userId)
112
- .where('status', 'active');
113
+ let countQuery = this._knex("chat")
114
+ .where("userId", userId)
115
+ .where("status", "active");
113
116
 
114
117
  // Apply search filter if query is provided
115
118
  if (query && query.trim()) {
116
119
  const searchTerm = `%${query.trim()}%`;
117
- dataQuery = dataQuery.where('title', 'ILIKE', searchTerm);
118
- countQuery = countQuery.where('title', 'ILIKE', searchTerm);
120
+ dataQuery = dataQuery.where("title", "ILIKE", searchTerm);
121
+ countQuery = countQuery.where("title", "ILIKE", searchTerm);
119
122
  }
120
123
 
121
124
  const [results, [{ count }]] = await Promise.all([
122
- dataQuery
123
- .orderBy('created_at', 'desc')
124
- .limit(limit)
125
- .offset(offset),
126
- countQuery.count('* as count')
125
+ dataQuery.orderBy("created_at", "desc").limit(limit).offset(offset),
126
+ countQuery.count("* as count"),
127
127
  ]);
128
128
 
129
129
  const totalCount = parseInt(count as string);
@@ -138,7 +138,7 @@ export class ChatDAO implements IBaseDAO<IChat> {
138
138
  count: results.length,
139
139
  totalCount,
140
140
  totalPages,
141
- hasMore
141
+ hasMore,
142
142
  };
143
143
  }
144
- }
144
+ }
@@ -2,71 +2,128 @@ import { Knex } from "knex";
2
2
  import { IBaseDAO, IDataPaginator } from "../../d.types";
3
3
  import { IFolder } from "../../interfaces/folder/folder.interfaces";
4
4
  import KnexManager from "../../KnexConnection";
5
+ import {
6
+ FolderSortField,
7
+ FolderStatus,
8
+ FOLDER_SORT_COLUMN_MAP,
9
+ } from "../../constants/folder.constants";
10
+ import { SortOrder } from "../../constants/video.constants";
11
+
12
+ export interface IFolderFilters {
13
+ studyId?: number | null;
14
+ status?: FolderStatus;
15
+ search?: string;
16
+ sortBy?: FolderSortField;
17
+ sortOrder?: SortOrder;
18
+ }
5
19
 
6
20
  export class FolderDAO implements IBaseDAO<IFolder> {
7
- private _knex: Knex<any, unknown[]> = KnexManager.getConnection();
21
+ private _knex: Knex<any, unknown[]> = KnexManager.getConnection();
8
22
 
9
- async create(item: IFolder): Promise<IFolder> {
10
- const [createdFolder] = await this._knex("folders").insert(item).returning("*");
11
- return createdFolder;
12
- }
23
+ async create(item: IFolder): Promise<IFolder> {
24
+ const [createdFolder] = await this._knex("folders")
25
+ .insert(item)
26
+ .returning("*");
27
+ return createdFolder;
28
+ }
13
29
 
14
- async getById(id: number): Promise<IFolder | null> {
15
- const folder = await this._knex("folders as f")
16
- .innerJoin("study as s", "f.studyId", "s.id")
17
- .select("f.*", this._knex.raw("to_jsonb(s.*) as study"))
18
- .where("f.id", id)
19
- .first();
20
- return folder || null;
21
- }
30
+ async getById(id: number): Promise<IFolder | null> {
31
+ const folder = await this._knex("folders as f")
32
+ .innerJoin("study as s", "f.studyId", "s.id")
33
+ .select("f.*", this._knex.raw("to_jsonb(s.*) as study"))
34
+ .where("f.id", id)
35
+ .first();
36
+ return folder || null;
37
+ }
22
38
 
23
- async getByUuid(uuid: string): Promise<IFolder | null> {
24
- const folder = await this._knex("folders as f")
25
- .innerJoin("study as s", "f.studyId", "s.id")
26
- .select("f.*", this._knex.raw("to_jsonb(s.*) as study"))
27
- .where("f.uuid", uuid)
28
- .first();
29
- return folder || null;
30
- }
39
+ async getByUuid(uuid: string): Promise<IFolder | null> {
40
+ const folder = await this._knex("folders as f")
41
+ .innerJoin("study as s", "f.studyId", "s.id")
42
+ .select("f.*", this._knex.raw("to_jsonb(s.*) as study"))
43
+ .where("f.uuid", uuid)
44
+ .first();
45
+ return folder || null;
46
+ }
31
47
 
32
- async update(id: number, item: Partial<IFolder>): Promise<IFolder | null> {
33
- const [updatedFolder] = await this._knex("folders").where({ id }).update(item).returning("*");
34
- return updatedFolder || null;
35
- }
48
+ async update(id: number, item: Partial<IFolder>): Promise<IFolder | null> {
49
+ const [updatedFolder] = await this._knex("folders")
50
+ .where({ id })
51
+ .update(item)
52
+ .returning("*");
53
+ return updatedFolder || null;
54
+ }
36
55
 
37
- async delete(id: number): Promise<boolean> {
38
- const result = await this._knex("folders").where({ id }).del();
39
- return result > 0;
40
- }
56
+ async delete(id: number): Promise<boolean> {
57
+ const result = await this._knex("folders").where({ id }).del();
58
+ return result > 0;
59
+ }
60
+
61
+ async getAll(
62
+ page: number,
63
+ limit: number,
64
+ filters?: IFolderFilters,
65
+ ): Promise<IDataPaginator<IFolder>> {
66
+ const offset = (page - 1) * limit;
67
+
68
+ const query = this._knex("folders as f")
69
+ .innerJoin("study as s", "f.studyId", "s.id")
70
+ .leftJoin("locations as l", "s.locationId", "l.id")
71
+ .select(
72
+ "f.*",
73
+ this._knex.raw("to_jsonb(s.*) as study"),
74
+ "l.name as locationName",
75
+ "l.uuid as locationUuid",
76
+ );
41
77
 
42
- async getAll(page: number, limit: number, studyId?: number | null): Promise<IDataPaginator<IFolder>> {
43
- const offset = (page - 1) * limit;
78
+ // Apply filters
79
+ if (filters) {
80
+ // Filter by studyId
81
+ if (filters.studyId !== undefined && filters.studyId !== null) {
82
+ query.where("f.studyId", filters.studyId);
83
+ }
44
84
 
45
- const query = this._knex("folders as f")
46
- .innerJoin("study as s", "f.studyId", "s.id")
47
- .leftJoin("locations as l", "s.locationId", "l.id")
48
- .select(
49
- "f.*",
50
- this._knex.raw("to_jsonb(s.*) as study"),
51
- "l.name as locationName",
52
- "l.uuid as locationUuid"
53
- );
54
- if (studyId !== undefined && studyId !== null) {
55
- query.where("f.studyId", studyId);
56
- }
85
+ // Filter by status
86
+ if (filters.status) {
87
+ query.where("f.status", filters.status);
88
+ }
57
89
 
58
- const [countResult] = await query.clone().clearSelect().count("* as count");
59
- const totalCount = +countResult.count;
60
- const folders = await query.clone().limit(limit).offset(offset);
90
+ // Search by name (case-insensitive)
91
+ if (filters.search) {
92
+ query.whereRaw("LOWER(f.name) LIKE ?", [
93
+ `%${filters.search.toLowerCase()}%`,
94
+ ]);
95
+ }
61
96
 
62
- return {
63
- success: true,
64
- data: folders,
65
- page,
66
- limit,
67
- count: folders.length,
68
- totalCount,
69
- totalPages: Math.ceil(totalCount / limit),
70
- };
97
+ // Apply sorting
98
+ if (filters.sortBy) {
99
+ const columnName = FOLDER_SORT_COLUMN_MAP[filters.sortBy];
100
+ const order = filters.sortOrder || "DESC";
101
+ query.orderBy(columnName, order);
102
+ } else {
103
+ // Default sort by created_at DESC
104
+ query.orderBy("f.created_at", "DESC");
105
+ }
106
+ } else {
107
+ // Default sort by created_at DESC
108
+ query.orderBy("f.created_at", "DESC");
71
109
  }
110
+
111
+ const [countResult] = await query
112
+ .clone()
113
+ .clearSelect()
114
+ .clearOrder()
115
+ .count("* as count");
116
+ const totalCount = +countResult.count;
117
+ const folders = await query.clone().limit(limit).offset(offset);
118
+
119
+ return {
120
+ success: true,
121
+ data: folders,
122
+ page,
123
+ limit,
124
+ count: folders.length,
125
+ totalCount,
126
+ totalPages: Math.ceil(totalCount / limit),
127
+ };
128
+ }
72
129
  }