user.queries.js 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. const { addMinutes } = require("date-fns");
  2. const { v4: uuid } = require("uuid");
  3. const { ROLES } = require("../consts");
  4. const utils = require("../utils");
  5. const redis = require("../redis");
  6. const knex = require("../knex");
  7. const env = require("../env");
  8. async function find(match) {
  9. if ((match.email || match.apikey) && env.REDIS_ENABLED) {
  10. const key = redis.key.user(match.email || match.apikey);
  11. const cachedUser = await redis.client.get(key);
  12. if (cachedUser) return JSON.parse(cachedUser);
  13. }
  14. const query = knex("users");
  15. Object.entries(match).forEach(([key, value]) => {
  16. query.andWhere(key, ...(Array.isArray(value) ? value : [value]));
  17. });
  18. const user = await query.first();
  19. if (user && env.REDIS_ENABLED) {
  20. const emailKey = redis.key.user(user.email);
  21. redis.client.set(emailKey, JSON.stringify(user), "EX", 60 * 15);
  22. if (user.apikey) {
  23. const apikeyKey = redis.key.user(user.apikey);
  24. redis.client.set(apikeyKey, JSON.stringify(user), "EX", 60 * 15);
  25. }
  26. }
  27. return user;
  28. }
  29. async function add(params, user) {
  30. const data = {
  31. email: params.email,
  32. password: params.password,
  33. ...(params.role && { role: params.role }),
  34. ...(params.verified !== undefined && { verified: params.verified }),
  35. verification_token: uuid(),
  36. verification_expires: utils.dateToUTC(addMinutes(new Date(), 60))
  37. };
  38. if (user) {
  39. await knex("users")
  40. .where("id", user.id)
  41. .update({ ...data, updated_at: utils.dateToUTC(new Date()) });
  42. } else {
  43. await knex("users").insert(data);
  44. }
  45. if (env.REDIS_ENABLED) {
  46. redis.remove.user(user);
  47. }
  48. return {
  49. ...user,
  50. ...data
  51. };
  52. }
  53. async function update(match, update, methods) {
  54. const query = knex("users");
  55. Object.entries(match).forEach(([key, value]) => {
  56. query.andWhere(key, ...(Array.isArray(value) ? value : [value]));
  57. });
  58. const updateQuery = query.clone();
  59. if (methods?.increments) {
  60. methods.increments.forEach(columnName => {
  61. updateQuery.increment(columnName);
  62. });
  63. }
  64. await updateQuery.update({ ...update, updated_at: utils.dateToUTC(new Date()) });
  65. const users = await query.select("*");
  66. if (env.REDIS_ENABLED) {
  67. users.forEach(redis.remove.user);
  68. }
  69. return users;
  70. }
  71. async function remove(user) {
  72. const deletedUser = await knex("users").where("id", user.id).delete();
  73. if (env.REDIS_ENABLED) {
  74. redis.remove.user(user);
  75. }
  76. return !!deletedUser;
  77. }
  78. const selectable_admin = [
  79. "users.id",
  80. "users.email",
  81. "users.verified",
  82. "users.role",
  83. "users.banned",
  84. "users.banned_by_id",
  85. "users.created_at",
  86. "users.updated_at"
  87. ];
  88. function normalizeMatch(match) {
  89. const newMatch = { ...match }
  90. if (newMatch.banned !== undefined) {
  91. newMatch["users.banned"] = newMatch.banned;
  92. delete newMatch.banned;
  93. }
  94. return newMatch;
  95. }
  96. async function getAdmin(match, params) {
  97. const query = knex("users")
  98. .select(...selectable_admin)
  99. .select("l.links_count")
  100. .select("d.domains")
  101. .fromRaw("users")
  102. .where(normalizeMatch(match))
  103. .offset(params.skip)
  104. .limit(params.limit)
  105. .orderBy("users.id", "desc")
  106. .groupBy(1)
  107. .groupBy("l.links_count")
  108. .groupBy("d.domains");
  109. if (params?.search) {
  110. const id = parseInt(params?.search);
  111. if (Number.isNaN(id)) {
  112. query.andWhereILike("users.email", "%" + params?.search + "%");
  113. } else {
  114. query.andWhere("users.id", params?.search);
  115. }
  116. }
  117. if (params?.domains !== undefined) {
  118. query.andWhere("d.domains", params?.domains ? "is not" : "is", null);
  119. }
  120. if (params?.links !== undefined) {
  121. query.andWhere("links_count", params?.links ? "is not" : "is", null);
  122. }
  123. query.leftJoin(
  124. knex("domains")
  125. .select("user_id", knex.raw("string_agg(address, ', ') AS domains"))
  126. .groupBy("user_id").as("d"),
  127. "users.id",
  128. "d.user_id"
  129. )
  130. query.leftJoin(
  131. knex("links").select("user_id").count("id as links_count").groupBy("user_id").as("l"),
  132. "users.id",
  133. "l.user_id"
  134. );
  135. return query;
  136. }
  137. async function totalAdmin(match, params) {
  138. const query = knex("users")
  139. .count("users.id")
  140. .fromRaw('users')
  141. .where(normalizeMatch(match));
  142. if (params?.search) {
  143. const id = parseInt(params?.search);
  144. if (Number.isNaN(id)) {
  145. query.andWhereILike("users.email", "%" + params?.search + "%");
  146. } else {
  147. query.andWhere("users.id", params?.search);
  148. }
  149. }
  150. if (params?.domains !== undefined) {
  151. query.andWhere("domains", params?.domains ? "is not" : "is", null);
  152. query.leftJoin(
  153. knex("domains")
  154. .select("user_id", knex.raw("string_agg(address, ', ') AS domains"))
  155. .groupBy("user_id").as("d"),
  156. "users.id",
  157. "d.user_id"
  158. );
  159. }
  160. if (params?.links !== undefined) {
  161. query.andWhere("links", params?.links ? "is not" : "is", null);
  162. query.leftJoin(
  163. knex("links").select("user_id").count("id as links").groupBy("user_id").as("l"),
  164. "users.id",
  165. "l.user_id"
  166. );
  167. }
  168. const [{count}] = await query;
  169. return typeof count === "number" ? count : parseInt(count);
  170. }
  171. async function create(params) {
  172. const [user] = await knex("users").insert({
  173. email: params.email,
  174. password: params.password,
  175. role: params.role ?? ROLES.USER,
  176. verified: params.verified ?? false,
  177. banned: params.banned ?? false,
  178. }, "*");
  179. return user;
  180. }
  181. // check if there exists a user
  182. async function findAny() {
  183. if (env.REDIS_ENABLED) {
  184. const anyuser = await redis.client.get("any-user");
  185. if (anyuser) return true;
  186. }
  187. const anyuser = await knex("users").select("id").first();
  188. if (env.REDIS_ENABLED && anyuser) {
  189. redis.client.set("any-user", JSON.stringify(anyuser), "EX", 60 * 5);
  190. }
  191. return !!anyuser;
  192. }
  193. module.exports = {
  194. add,
  195. create,
  196. find,
  197. findAny,
  198. getAdmin,
  199. remove,
  200. totalAdmin,
  201. update,
  202. }