user.queries.js 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  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.id || match.apikey) && env.REDIS_ENABLED) {
  10. const key = redis.key.user(match.id || 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 idKey = redis.key.user(user.id);
  21. redis.client.set(idKey, 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 { user, updated_user } = await knex.transaction(async function(trx) {
  55. const query = trx("users");
  56. Object.entries(match).forEach(([key, value]) => {
  57. query.andWhere(key, ...(Array.isArray(value) ? value : [value]));
  58. });
  59. const user = await query.select("id").first();
  60. if (!user) return null;
  61. const updateQuery = trx("users").where("id", user.id);
  62. if (methods?.increments) {
  63. methods.increments.forEach(columnName => {
  64. updateQuery.increment(columnName);
  65. });
  66. }
  67. await updateQuery.update({ ...update, updated_at: utils.dateToUTC(new Date()) });
  68. const updated_user = await trx("users").where("id", user.id).first();
  69. return { user, updated_user };
  70. });
  71. if (env.REDIS_ENABLED && user) {
  72. redis.remove.user(user);
  73. redis.remove.user(updated_user);
  74. }
  75. return updated_user;
  76. }
  77. async function remove(user) {
  78. const deletedUser = await knex("users").where("id", user.id).delete();
  79. if (env.REDIS_ENABLED) {
  80. redis.remove.user(user);
  81. }
  82. return !!deletedUser;
  83. }
  84. const selectable_admin = [
  85. "users.id",
  86. "users.email",
  87. "users.verified",
  88. "users.role",
  89. "users.banned",
  90. "users.banned_by_id",
  91. "users.created_at",
  92. "users.updated_at"
  93. ];
  94. function normalizeMatch(match) {
  95. const newMatch = { ...match }
  96. if (newMatch.banned !== undefined) {
  97. newMatch["users.banned"] = newMatch.banned;
  98. delete newMatch.banned;
  99. }
  100. return newMatch;
  101. }
  102. async function getAdmin(match, params) {
  103. const query = knex("users")
  104. .select(...selectable_admin)
  105. .select("l.links_count")
  106. .select("d.domains")
  107. .fromRaw("users")
  108. .where(normalizeMatch(match))
  109. .offset(params.skip)
  110. .limit(params.limit)
  111. .orderBy("users.id", "desc")
  112. .groupBy(1)
  113. .groupBy("l.links_count")
  114. .groupBy("d.domains");
  115. if (params?.search) {
  116. const id = parseInt(params?.search);
  117. if (Number.isNaN(id)) {
  118. query.andWhereILike("users.email", "%" + params?.search + "%");
  119. } else {
  120. query.andWhere("users.id", params?.search);
  121. }
  122. }
  123. if (params?.domains !== undefined) {
  124. query.andWhere("d.domains", params?.domains ? "is not" : "is", null);
  125. }
  126. if (params?.links !== undefined) {
  127. query.andWhere("links_count", params?.links ? "is not" : "is", null);
  128. }
  129. query.leftJoin(
  130. knex("domains")
  131. .select("user_id", knex.raw("string_agg(address, ', ') AS domains"))
  132. .groupBy("user_id").as("d"),
  133. "users.id",
  134. "d.user_id"
  135. )
  136. query.leftJoin(
  137. knex("links").select("user_id").count("id as links_count").groupBy("user_id").as("l"),
  138. "users.id",
  139. "l.user_id"
  140. );
  141. return query;
  142. }
  143. async function totalAdmin(match, params) {
  144. const query = knex("users")
  145. .count("users.id as count")
  146. .fromRaw('users')
  147. .where(normalizeMatch(match));
  148. if (params?.search) {
  149. const id = parseInt(params?.search);
  150. if (Number.isNaN(id)) {
  151. query.andWhereILike("users.email", "%" + params?.search + "%");
  152. } else {
  153. query.andWhere("users.id", params?.search);
  154. }
  155. }
  156. if (params?.domains !== undefined) {
  157. query.andWhere("domains", params?.domains ? "is not" : "is", null);
  158. query.leftJoin(
  159. knex("domains")
  160. .select("user_id", knex.raw("string_agg(address, ', ') AS domains"))
  161. .groupBy("user_id").as("d"),
  162. "users.id",
  163. "d.user_id"
  164. );
  165. }
  166. if (params?.links !== undefined) {
  167. query.andWhere("links", params?.links ? "is not" : "is", null);
  168. query.leftJoin(
  169. knex("links").select("user_id").count("id as links").groupBy("user_id").as("l"),
  170. "users.id",
  171. "l.user_id"
  172. );
  173. }
  174. const [{count}] = await query;
  175. return typeof count === "number" ? count : parseInt(count);
  176. }
  177. async function create(params) {
  178. let [user] = await knex("users").insert({
  179. email: params.email,
  180. password: params.password,
  181. role: params.role ?? ROLES.USER,
  182. verified: params.verified ?? false,
  183. banned: params.banned ?? false,
  184. }, "*");
  185. // mysql doesn't return the whole user, but rather the id number only
  186. // so we need to fetch the user ourselves
  187. if (typeof user === "number") {
  188. user = await knex("users").where("id", user).first();
  189. }
  190. return user;
  191. }
  192. // check if there exists a user
  193. async function findAny() {
  194. if (env.REDIS_ENABLED) {
  195. const anyuser = await redis.client.get("any-user");
  196. if (anyuser) return true;
  197. }
  198. const anyuser = await knex("users").select("id").first();
  199. if (env.REDIS_ENABLED && anyuser) {
  200. redis.client.set("any-user", JSON.stringify(anyuser), "EX", 60 * 5);
  201. }
  202. return !!anyuser;
  203. }
  204. module.exports = {
  205. add,
  206. create,
  207. find,
  208. findAny,
  209. getAdmin,
  210. remove,
  211. totalAdmin,
  212. update,
  213. }