@nlabs/reaktor 0.5.1 → 0.5.2
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.
|
@@ -329,4 +329,4 @@ const deleteUserFromConversation = (context, convoId, userId) => {
|
|
|
329
329
|
getDirectConversationOld,
|
|
330
330
|
updateConversation
|
|
331
331
|
});
|
|
332
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/conversations.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseArangoId, parseId} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport cloneDeep from 'lodash/cloneDeep';\nimport isEmpty from 'lodash/isEmpty';\nimport {ApiContext} from 'types/auth';\n\nimport {ArangoDBLimit, ConnectionType, ConversationType, ImageUrlData, User} from '../types';\nimport {getLimit} from '../utils';\nimport {logError} from '../utils/analytics';\nimport {getUserImageUrl} from './images';\nimport {getDisplayName, getUser} from './users';\n\nconst eventCategory: string = 'conversations';\n\nexport const getConversations = (context: ApiContext, {from, to}): Promise<ConversationType[]> => {\n  const action: string = 'getConversations';\n  const {database, session: {userId: sessionId}} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const sessionDocId: string = `users/${sessionId}`;\n  const aqlQry: string = `FOR c, e IN 1..1 OUTBOUND \"${sessionDocId}\" hasConversations\n    ${limit.aql}\n    LET users = (\n      FOR u, uc IN 1..1 INBOUND c._id hasConversations\n      FILTER u._id != \"${sessionDocId}\"\n      RETURN u\n    )\n    RETURN MERGE(c, {users: users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDirectConversation = async (\n  context: ApiContext,\n  fromId: string,\n  toId: string\n): Promise<ConversationType> => {\n  const action: string = 'getDirectConversation';\n  const {database} = context;\n\n  try {\n    const formatFromId: string = parseId(fromId);\n    const formatToId: string = parseId(toId);\n    const userIds: string[] = [formatFromId, formatToId];\n    const sortedUserIds: string[] = userIds.sort((a, b) => {\n      if(a < b) {\n        return -1;\n      }\n      if(a > b) {\n        return 1;\n      }\n      return 0;\n    });\n    const conversationId: string = createHash(`conversations-${sortedUserIds.join('-')}`, null);\n    const user: User = await getUser(context, {userId: formatToId});\n\n    const aqlQry: AqlQuery = aql`LET c = DOCUMENT(${`conversations/${conversationId}`}) RETURN c`;\n    const conversation = await database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, error, context).then(() => null));\n\n    if(conversation) {\n      return {...conversation, users: [user]};\n    }\n\n    const now: number = Date.now();\n    const insert = {\n      _key: conversationId,\n      added: now,\n      isDirect: true,\n      modified: now\n    };\n    const insertQuery: AqlQuery = aql`INSERT ${insert} IN conversations RETURN NEW`;\n    const updatedConversation = await database.query(insertQuery)\n      .then((cursor: ArrayCursor) => cursor.next());\n\n    await Promise.all(userIds.map((userId) => {\n      const edgeId: string = createHash(`hasConversations-${userId}-${conversationId}`);\n      const edge: any = {\n        _from: `users/${userId}`,\n        _key: edgeId,\n        _to: `conversations/${conversationId}`,\n        added: Date.now()\n      };\n      const insertEdgeQuery: AqlQuery = aql`INSERT ${edge} IN hasConversations RETURN NEW`;\n      return database.query(insertEdgeQuery);\n    }));\n\n    return {...updatedConversation, users: [user]};\n  } catch(updateError) {\n    return logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, updateError, context).then(() => null);\n  }\n};\n\nexport const getConnectionUsers = (\n  context: ApiContext,\n  conversationId: string,\n  includeAll: boolean = false,\n  isDirect?: boolean\n): Promise<ConnectionType[]> => {\n  const action: string = 'getConversationUsers';\n  const {database, session: {userId: sessionUserId}} = context;\n  const filters: string[] = ['c.userId == u._key'];\n\n  if(!includeAll) {\n    filters.push(`u._key != ${sessionUserId}`);\n  }\n  if(isDirect) {\n    filters.push(`p.vertices[0].isDirect == ${isDirect}`);\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations\n    OPTIONS {vertexCollections: \"users\"}\n    FOR c IN connections\n    FILTER ${filters.join(' && ')}\n    RETURN c`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getConversationUsers = (\n  context: ApiContext,\n  conversationId: string,\n  includeAll: boolean = false,\n  isDirect?: boolean\n): Promise<ConversationType[]> => {\n  const action: string = 'getConversationUsers';\n  const {database, session: {userId: sessionUserId}} = context;\n  const filters: string[] = ['c.userId == u._key'];\n\n  if(!includeAll) {\n    filters.push(`u._key != ${sessionUserId}`);\n  }\n  if(isDirect) {\n    filters.push(`p.vertices[0].isDirect == ${isDirect}`);\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations\n    OPTIONS {vertexCollections: \"users\"}\n    FILTER ${filters.join(' && ')}\n    RETURN u`;\n\n  console.log({aqlQry});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDirectConversationOld = (context: ApiContext, userId: string): Promise<ConversationType> => {\n  const action: string = 'getDirectConversation';\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${sessionId}`;\n  const formatUserId: string = parseArangoId(userId);\n\n  const aqlQry: AqlQuery = aql`LET from = (\n      FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n      FILTER c.direct == true\n      RETURN c\n    )\n    LET to = (\n      FOR c, e IN 1..1 OUTBOUND ${formatUserId} hasConversations\n      FILTER c.direct == true\n      RETURN c\n    )\n    LET conversation = FIRST(INTERSECTION(from, to))\n    LET users = !IS_NULL(conversation) && (\n      FOR g IN[${sessionDocId}, ${formatUserId}]\n      FOR u IN users\n      FILTER g == u._id\n      RETURN u\n    )\n    RETURN IS_NULL(conversation) ? null : MERGE(conversation, {users: users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getConversation = (context: ApiContext, convoId: string): Promise<ConversationType> => {\n  const action: string = 'getItem';\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${sessionId}`;\n  const convoDocId: string = `conversations/${parseId(convoId)}`;\n\n  const aqlQry: AqlQuery = aql`FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n    FILTER c._id == ${convoDocId}\n    LET u = (\n      FOR inUser, inEdge IN 1..1 INBOUND ${convoDocId} hasConversations\n      RETURN inUser\n    )\n    LIMIT 1\n    RETURN MERGE(c, {users: u})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversation: ConversationType = {}) => {\n      if(!isEmpty(conversation)) {\n        conversation.users = conversation.users.map((user: any) => {\n          const {imageId, userId} = user;\n          const thumbUrlData: ImageUrlData = {\n            imageId,\n            isThumb: true,\n            userId\n          };\n          return {\n            ...user,\n            name: getDisplayName(user),\n            thumb: getUserImageUrl(thumbUrlData)\n          };\n        });\n\n        return conversation;\n      }\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const updateConversation = (context: ApiContext, conversation): Promise<ConversationType> => {\n  const action: string = 'updateConversation';\n  const {database, session: {userId: sessionId}} = context;\n  const {\n    conversationId = createHash(`conversation-${sessionId} `),\n    direct = false,\n    name\n  } = conversation;\n  const formatConversationId: string = parseId(conversationId);\n  const now: number = Date.now();\n  const update: any = {\n    // Direct message\n    direct,\n    modified: now,\n    name\n  };\n\n  const insert: any = {\n    ...cloneDeep(update),\n    _key: formatConversationId,\n    added: now\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${formatConversationId} }\n    INSERT ${insert}\n    UPDATE ${update}\n    IN conversations RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversation: ConversationType = {}) => conversation)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const addUserToConversation = (context: ApiContext, convoId: string, userId: string) => {\n  const action: string = 'addUserToConversation';\n  const {database} = context;\n  const formatConvoId: string = parseId(convoId);\n  const formatUserId: string = parseId(userId);\n  const convoDocId: string = `conversations/${formatConvoId} `;\n  const userDocId: string = `users/${formatUserId} `;\n\n  const aqlQry: AqlQuery = aql`FOR c IN hasConversations\n    FILTER c._from == ${userDocId} && c._to == ${convoDocId}\n    LIMIT 1\n    RETURN c`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversationEdge) => {\n      if(!!conversationEdge) {\n        return conversationEdge;\n      }\n\n      const edgeCollection: EdgeCollection = database.collection('hasConversations');\n      const edgeId: string = createHash(`convo - ${formatConvoId} -${formatUserId} `);\n\n      const edge: ConversationType = {\n        _from: userDocId,\n        _key: edgeId,\n        _to: convoDocId,\n        added: Date.now()\n      };\n\n      return edgeCollection.save(edge, {returnNew: true}).then(() => edge);\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteUserFromConversation = (context: ApiContext, convoId: string, userId: string) => {\n  const action: string = 'deleteUserFromConversation';\n  const {database} = context;\n  const formatConvoId: string = parseId(convoId);\n  const formatUserId: string = parseId(userId);\n  const convoDocId: string = `conversations / ${formatConvoId} `;\n  const userDocId: string = `users/${formatUserId} `;\n\n  const aqlQry: AqlQuery = aql`FOR c IN hasConversations\n    FILTER c._from == ${userDocId} && c._to == ${convoDocId}\n    LIMIT 1\n    REMOVE c IN hasConversations\n    RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n"],
  "mappings": "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",
  "names": []
}

|
|
332
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/conversations.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {createHash, parseArangoId, parseId} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport cloneDeep from 'lodash/cloneDeep';\nimport isEmpty from 'lodash/isEmpty';\n\nimport {ArangoDBLimit, ConnectionType, ConversationType, ImageUrlData, User} from '../types';\nimport {ApiContext} from '../types/auth';\nimport {getLimit} from '../utils';\nimport {logError} from '../utils/analytics';\nimport {getUserImageUrl} from './images';\nimport {getDisplayName, getUser} from './users';\n\nconst eventCategory: string = 'conversations';\n\nexport const getConversations = (context: ApiContext, {from, to}): Promise<ConversationType[]> => {\n  const action: string = 'getConversations';\n  const {database, session: {userId: sessionId}} = context;\n  const limit: ArangoDBLimit = getLimit(from, to);\n  const sessionDocId: string = `users/${sessionId}`;\n  const aqlQry: string = `FOR c, e IN 1..1 OUTBOUND \"${sessionDocId}\" hasConversations\n    ${limit.aql}\n    LET users = (\n      FOR u, uc IN 1..1 INBOUND c._id hasConversations\n      FILTER u._id != \"${sessionDocId}\"\n      RETURN u\n    )\n    RETURN MERGE(c, {users: users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDirectConversation = async (\n  context: ApiContext,\n  fromId: string,\n  toId: string\n): Promise<ConversationType> => {\n  const action: string = 'getDirectConversation';\n  const {database} = context;\n\n  try {\n    const formatFromId: string = parseId(fromId);\n    const formatToId: string = parseId(toId);\n    const userIds: string[] = [formatFromId, formatToId];\n    const sortedUserIds: string[] = userIds.sort((a, b) => {\n      if(a < b) {\n        return -1;\n      }\n      if(a > b) {\n        return 1;\n      }\n      return 0;\n    });\n    const conversationId: string = createHash(`conversations-${sortedUserIds.join('-')}`, null);\n    const user: User = await getUser(context, {userId: formatToId});\n\n    const aqlQry: AqlQuery = aql`LET c = DOCUMENT(${`conversations/${conversationId}`}) RETURN c`;\n    const conversation = await database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, error, context).then(() => null));\n\n    if(conversation) {\n      return {...conversation, users: [user]};\n    }\n\n    const now: number = Date.now();\n    const insert = {\n      _key: conversationId,\n      added: now,\n      isDirect: true,\n      modified: now\n    };\n    const insertQuery: AqlQuery = aql`INSERT ${insert} IN conversations RETURN NEW`;\n    const updatedConversation = await database.query(insertQuery)\n      .then((cursor: ArrayCursor) => cursor.next());\n\n    await Promise.all(userIds.map((userId) => {\n      const edgeId: string = createHash(`hasConversations-${userId}-${conversationId}`);\n      const edge: any = {\n        _from: `users/${userId}`,\n        _key: edgeId,\n        _to: `conversations/${conversationId}`,\n        added: Date.now()\n      };\n      const insertEdgeQuery: AqlQuery = aql`INSERT ${edge} IN hasConversations RETURN NEW`;\n      return database.query(insertEdgeQuery);\n    }));\n\n    return {...updatedConversation, users: [user]};\n  } catch(updateError) {\n    return logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, updateError, context).then(() => null);\n  }\n};\n\nexport const getConnectionUsers = (\n  context: ApiContext,\n  conversationId: string,\n  includeAll: boolean = false,\n  isDirect?: boolean\n): Promise<ConnectionType[]> => {\n  const action: string = 'getConversationUsers';\n  const {database, session: {userId: sessionUserId}} = context;\n  const filters: string[] = ['c.userId == u._key'];\n\n  if(!includeAll) {\n    filters.push(`u._key != ${sessionUserId}`);\n  }\n  if(isDirect) {\n    filters.push(`p.vertices[0].isDirect == ${isDirect}`);\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations\n    OPTIONS {vertexCollections: \"users\"}\n    FOR c IN connections\n    FILTER ${filters.join(' && ')}\n    RETURN c`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getConversationUsers = (\n  context: ApiContext,\n  conversationId: string,\n  includeAll: boolean = false,\n  isDirect?: boolean\n): Promise<ConversationType[]> => {\n  const action: string = 'getConversationUsers';\n  const {database, session: {userId: sessionUserId}} = context;\n  const filters: string[] = ['c.userId == u._key'];\n\n  if(!includeAll) {\n    filters.push(`u._key != ${sessionUserId}`);\n  }\n  if(isDirect) {\n    filters.push(`p.vertices[0].isDirect == ${isDirect}`);\n  }\n\n  const aqlQry: AqlQuery = aql`FOR u, e,p IN 1..1 INBOUND ${`conversations/${conversationId}`} hasConversations\n    OPTIONS {vertexCollections: \"users\"}\n    FILTER ${filters.join(' && ')}\n    RETURN u`;\n\n  console.log({aqlQry});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getDirectConversationOld = (context: ApiContext, userId: string): Promise<ConversationType> => {\n  const action: string = 'getDirectConversation';\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${sessionId}`;\n  const formatUserId: string = parseArangoId(userId);\n\n  const aqlQry: AqlQuery = aql`LET from = (\n      FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n      FILTER c.direct == true\n      RETURN c\n    )\n    LET to = (\n      FOR c, e IN 1..1 OUTBOUND ${formatUserId} hasConversations\n      FILTER c.direct == true\n      RETURN c\n    )\n    LET conversation = FIRST(INTERSECTION(from, to))\n    LET users = !IS_NULL(conversation) && (\n      FOR g IN[${sessionDocId}, ${formatUserId}]\n      FOR u IN users\n      FILTER g == u._id\n      RETURN u\n    )\n    RETURN IS_NULL(conversation) ? null : MERGE(conversation, {users: users})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const getConversation = (context: ApiContext, convoId: string): Promise<ConversationType> => {\n  const action: string = 'getItem';\n  const {database, session: {userId: sessionId}} = context;\n  const sessionDocId: string = `users/${sessionId}`;\n  const convoDocId: string = `conversations/${parseId(convoId)}`;\n\n  const aqlQry: AqlQuery = aql`FOR c, e IN 1..1 OUTBOUND ${sessionDocId} hasConversations\n    FILTER c._id == ${convoDocId}\n    LET u = (\n      FOR inUser, inEdge IN 1..1 INBOUND ${convoDocId} hasConversations\n      RETURN inUser\n    )\n    LIMIT 1\n    RETURN MERGE(c, {users: u})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversation: ConversationType = {}) => {\n      if(!isEmpty(conversation)) {\n        conversation.users = conversation.users.map((user: any) => {\n          const {imageId, userId} = user;\n          const thumbUrlData: ImageUrlData = {\n            imageId,\n            isThumb: true,\n            userId\n          };\n          return {\n            ...user,\n            name: getDisplayName(user),\n            thumb: getUserImageUrl(thumbUrlData)\n          };\n        });\n\n        return conversation;\n      }\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const updateConversation = (context: ApiContext, conversation): Promise<ConversationType> => {\n  const action: string = 'updateConversation';\n  const {database, session: {userId: sessionId}} = context;\n  const {\n    conversationId = createHash(`conversation-${sessionId} `),\n    direct = false,\n    name\n  } = conversation;\n  const formatConversationId: string = parseId(conversationId);\n  const now: number = Date.now();\n  const update: any = {\n    // Direct message\n    direct,\n    modified: now,\n    name\n  };\n\n  const insert: any = {\n    ...cloneDeep(update),\n    _key: formatConversationId,\n    added: now\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${formatConversationId} }\n    INSERT ${insert}\n    UPDATE ${update}\n    IN conversations RETURN NEW`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversation: ConversationType = {}) => conversation)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const addUserToConversation = (context: ApiContext, convoId: string, userId: string) => {\n  const action: string = 'addUserToConversation';\n  const {database} = context;\n  const formatConvoId: string = parseId(convoId);\n  const formatUserId: string = parseId(userId);\n  const convoDocId: string = `conversations/${formatConvoId} `;\n  const userDocId: string = `users/${formatUserId} `;\n\n  const aqlQry: AqlQuery = aql`FOR c IN hasConversations\n    FILTER c._from == ${userDocId} && c._to == ${convoDocId}\n    LIMIT 1\n    RETURN c`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((conversationEdge) => {\n      if(!!conversationEdge) {\n        return conversationEdge;\n      }\n\n      const edgeCollection: EdgeCollection = database.collection('hasConversations');\n      const edgeId: string = createHash(`convo - ${formatConvoId} -${formatUserId} `);\n\n      const edge: ConversationType = {\n        _from: userDocId,\n        _key: edgeId,\n        _to: convoDocId,\n        added: Date.now()\n      };\n\n      return edgeCollection.save(edge, {returnNew: true}).then(() => edge);\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n\nexport const deleteUserFromConversation = (context: ApiContext, convoId: string, userId: string) => {\n  const action: string = 'deleteUserFromConversation';\n  const {database} = context;\n  const formatConvoId: string = parseId(convoId);\n  const formatUserId: string = parseId(userId);\n  const convoDocId: string = `conversations / ${formatConvoId} `;\n  const userDocId: string = `users/${formatUserId} `;\n\n  const aqlQry: AqlQuery = aql`FOR c IN hasConversations\n    FILTER c._from == ${userDocId} && c._to == ${convoDocId}\n    LIMIT 1\n    REMOVE c IN hasConversations\n    RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => null));\n};\n"],
  "mappings": "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",
  "names": []
}

|
package/lib/actions/files.d.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
/// <reference types="node" />
|
|
2
2
|
import { Database } from 'arangojs';
|
|
3
|
-
import { ApiContext } from 'types/auth';
|
|
3
|
+
import { ApiContext } from '../types/auth';
|
|
4
4
|
import { FileType } from '../types/files';
|
|
5
5
|
export declare const addFile: (context: ApiContext, item?: FileType) => Promise<FileType>;
|
|
6
6
|
export declare const getGiphyTrends: (context: ApiContext, limit?: number) => Promise<any[]>;
|
package/lib/actions/files.js
CHANGED
|
@@ -316,4 +316,4 @@ const decodeBase64 = (dataString) => {
|
|
|
316
316
|
linkFiles,
|
|
317
317
|
updateFiles
|
|
318
318
|
});
|
|
319
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/files.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {get as httpGet} from '@nlabs/rip-hunter';\nimport {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {google} from 'googleapis';\nimport {UserError} from 'graphql-errors';\nimport isEmpty from 'lodash/isEmpty';\nimport * as request from 'request-promise';\nimport {ApiContext} from 'types/auth';\n\nimport {Config} from '../config';\nimport {FileType} from '../types/files';\nimport {resizeSaveImage} from './images';\nimport {createPostEdge} from './posts';\n\nconst youtube = google.youtube({auth: Config.get('google.key'), version: 'v3'});\nrequest.defaults({encoding: null});\n\n// const eventCategory: string = 'files';\n\n// Upload file\nexport const addFile = (context: ApiContext, item: FileType = {}): Promise<FileType> => {\n  const {database, session: {userId: sessionId, userAccess}} = context;\n  const {\n    description,\n    fileId,\n    fileType,\n    name,\n    url\n  } = item;\n\n  // Id\n  const formatFileId: string = fileId ? parseId(fileId) : createHash(`file-${sessionId}`);\n\n  // Name\n  const isUrl: boolean = isEmpty(url);\n\n  // If no name, get it from url path\n  let formatName: string = parseString(name, 160);\n  let formatType: string = parseString(fileType, 16);\n\n  if(formatName === '' && isUrl) {\n    formatName = url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  if(formatType === '') {\n    const nameArr: string[] = formatName.split('.');\n    const ext: string = nameArr[nameArr.length - 1];\n\n    switch(ext) {\n      case 'jpeg':\n      case 'jpg':\n        formatType = 'image/jpeg';\n        break;\n      case 'png':\n        formatType = 'image/png';\n        break;\n      case 'zip':\n        formatType = 'application/zip';\n        break;\n      default:\n        break;\n    }\n  }\n\n  let isImage: boolean;\n\n  switch(formatType) {\n    case 'image/jpeg':\n    case 'image/png':\n      isImage = true;\n      break;\n    default:\n      isImage = false;\n      break;\n  }\n\n  // Description\n  const formatDesc: string = parseString(description, 500);\n\n  // Only allow file uploads to premium users\n  if(!isImage && userAccess !== 2) {\n    throw new UserError('account_restriction');\n  }\n\n  const saveToDb = (insert: FileType) => {\n    const aqlQry: AqlQuery = aql`INSERT ${insert} IN files RETURN NEW`;\n\n    return database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((file = {}) => file)\n      .catch((error: Error) => {\n        throw error;\n      });\n  };\n\n  const uploadFile = (buf: Buffer, uploadType: string) => {\n    const now: number = Date.now();\n\n    // If image, resize and create a thumbnail\n    if(isImage) {\n      return resizeSaveImage(context, formatFileId, buf, uploadType)\n        .then((resizedImage: FileType) => {\n          const insert: FileType = {\n            ...resizedImage,\n            _key: formatFileId,\n            added: now,\n            description: formatDesc,\n            fileType: formatType,\n            modified: now,\n            name: formatName,\n            userId: sessionId\n          };\n\n          return saveToDb(insert);\n        })\n        .catch((error: Error) => {\n          throw error;\n        });\n    }\n    const insert: FileType = {\n      _key: formatFileId,\n      added: now,\n      description: formatDesc,\n      fileType: formatType,\n      modified: now,\n      name: formatName,\n      userId: sessionId\n    };\n\n    return saveToDb(insert);\n  };\n\n  // If file is a url path, download the file and save\n  if(isUrl) {\n    return request.get({encoding: null, uri: url})\n      .then((body) => uploadFile(Buffer.from(body, 'binary'), formatType))\n      .catch(() => {\n        throw new UserError('file_request');\n      });\n  } else if(item.base64 !== '') {\n    const buffer: Buffer = Buffer.from(item.base64);\n    return uploadFile(buffer, formatType);\n  }\n  throw new Error('file_required');\n};\n\n// Giphy\nexport const getGiphyTrends = (context: ApiContext, limit: number = 30): Promise<any[]> => {\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/trending?api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return httpGet(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getGiphySearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> => {\n  const formatQuery: string = encodeURI(query);\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return fetch(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getYouTubeTrends = (context: ApiContext, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.videos.list({\n      chart: 'mostPopular',\n      maxResults: limit,\n      part: ['snippet'],\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const list = data.items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n\nexport const getYouTubeSearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.search.list({\n      maxResults: limit,\n      part: ['snippet'],\n      // eslint-disable-next-line\n      q: query,\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const {items} = data;\n        const list = items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n// Files\nexport const getPathUserFiles = (userId: string, filename: string): string => `users/${userId}/files/${filename}`;\n\nexport const getUrlUserFiles = (userId: string, filename: string, dir: string = 'files', type: string = 'profile'): string => {\n  if(filename) {\n    return `https://box.${Config.get('app.url')}/users/${userId}/${dir}/${filename}`;\n  }\n\n  if(type === 'profile') {\n    return `https://box.${Config.get('app.url')}/defaults/user_bk.jpg`;\n  }\n\n  return `https://box.${Config.get('app.url')}/defaults/user_wh.jpg`;\n};\n\nexport const createFile = (db: Database, file: FileType): Promise<FileType> => {\n  const {fileId} = file;\n  const insert: any = {\n    _key: fileId,\n    added: Date.now()\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${fileId}}\n    INSERT ${insert}\n    UPDATE {}\n    IN files RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((updatedFile: FileType = {}) => updatedFile)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const linkFiles = (db: Database, files: FileType[], postId: string): Promise<any> =>\n  Promise.all(\n    files.map((file: FileType) => createFile(db, file)\n      .then((file: FileType) => createPostEdge(db, file, postId)))\n  );\n\nexport const updateFiles = (db: Database, postId: string, files: FileType[]): Promise<any> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n\n  return edgeCollection.inEdges(postId)\n    .then((edges: any) => {\n      if(edges.length) {\n        // Remove linked edges\n        return Promise.all(\n          edges.map((edge) => {\n            const {_key: edgeKey} = edge;\n            const aqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN isPosted`;\n\n            return db.query(aqlQry).catch((error: Error) => {\n              throw error;\n            });\n          }))\n          .then(() => {\n            if(files.length) {\n              // Link files\n              return linkFiles(db, files, postId).then(() => files);\n            }\n            return files;\n          });\n      } else if(files.length) {\n        // Link files\n        return linkFiles(db, files, postId).then(() => files);\n      }\n      return files;\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const encodeBase64 = (buffer: Buffer): string => Buffer.from(buffer).toString('base64');\n\nexport const decodeBase64 = (dataString: string): object => {\n  const getData = (str: string) => str.match(/^data:([A-Za-z-+\\/]+);base64,(.+)$/) || [];\n  // const getData = (base64: string) => base64.substr(base64.indexOf(',') + 1);\n  let matches = getData(dataString);\n\n  if(matches.length !== 3) {\n    // If invalid make sure we don't need to decode\n    matches = getData(decodeURIComponent(dataString));\n\n    // Check it again.\n    if(matches.length !== 3) {\n      throw Error('Invalid input string');\n    }\n  }\n\n  return {\n    data: Buffer.from(matches[2], 'base64'),\n    type: matches[1]\n  };\n};\n"],
  "mappings": "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",
  "names": []
}

|
|
319
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/files.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {get as httpGet} from '@nlabs/rip-hunter';\nimport {createHash, parseId, parseString} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {google} from 'googleapis';\nimport {UserError} from 'graphql-errors';\nimport isEmpty from 'lodash/isEmpty';\nimport * as request from 'request-promise';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {FileType} from '../types/files';\nimport {resizeSaveImage} from './images';\nimport {createPostEdge} from './posts';\n\nconst youtube = google.youtube({auth: Config.get('google.key'), version: 'v3'});\nrequest.defaults({encoding: null});\n\n// const eventCategory: string = 'files';\n\n// Upload file\nexport const addFile = (context: ApiContext, item: FileType = {}): Promise<FileType> => {\n  const {database, session: {userId: sessionId, userAccess}} = context;\n  const {\n    description,\n    fileId,\n    fileType,\n    name,\n    url\n  } = item;\n\n  // Id\n  const formatFileId: string = fileId ? parseId(fileId) : createHash(`file-${sessionId}`);\n\n  // Name\n  const isUrl: boolean = isEmpty(url);\n\n  // If no name, get it from url path\n  let formatName: string = parseString(name, 160);\n  let formatType: string = parseString(fileType, 16);\n\n  if(formatName === '' && isUrl) {\n    formatName = url.substring(url.lastIndexOf('/') + 1);\n  }\n\n  if(formatType === '') {\n    const nameArr: string[] = formatName.split('.');\n    const ext: string = nameArr[nameArr.length - 1];\n\n    switch(ext) {\n      case 'jpeg':\n      case 'jpg':\n        formatType = 'image/jpeg';\n        break;\n      case 'png':\n        formatType = 'image/png';\n        break;\n      case 'zip':\n        formatType = 'application/zip';\n        break;\n      default:\n        break;\n    }\n  }\n\n  let isImage: boolean;\n\n  switch(formatType) {\n    case 'image/jpeg':\n    case 'image/png':\n      isImage = true;\n      break;\n    default:\n      isImage = false;\n      break;\n  }\n\n  // Description\n  const formatDesc: string = parseString(description, 500);\n\n  // Only allow file uploads to premium users\n  if(!isImage && userAccess !== 2) {\n    throw new UserError('account_restriction');\n  }\n\n  const saveToDb = (insert: FileType) => {\n    const aqlQry: AqlQuery = aql`INSERT ${insert} IN files RETURN NEW`;\n\n    return database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((file = {}) => file)\n      .catch((error: Error) => {\n        throw error;\n      });\n  };\n\n  const uploadFile = (buf: Buffer, uploadType: string) => {\n    const now: number = Date.now();\n\n    // If image, resize and create a thumbnail\n    if(isImage) {\n      return resizeSaveImage(context, formatFileId, buf, uploadType)\n        .then((resizedImage: FileType) => {\n          const insert: FileType = {\n            ...resizedImage,\n            _key: formatFileId,\n            added: now,\n            description: formatDesc,\n            fileType: formatType,\n            modified: now,\n            name: formatName,\n            userId: sessionId\n          };\n\n          return saveToDb(insert);\n        })\n        .catch((error: Error) => {\n          throw error;\n        });\n    }\n    const insert: FileType = {\n      _key: formatFileId,\n      added: now,\n      description: formatDesc,\n      fileType: formatType,\n      modified: now,\n      name: formatName,\n      userId: sessionId\n    };\n\n    return saveToDb(insert);\n  };\n\n  // If file is a url path, download the file and save\n  if(isUrl) {\n    return request.get({encoding: null, uri: url})\n      .then((body) => uploadFile(Buffer.from(body, 'binary'), formatType))\n      .catch(() => {\n        throw new UserError('file_request');\n      });\n  } else if(item.base64 !== '') {\n    const buffer: Buffer = Buffer.from(item.base64);\n    return uploadFile(buffer, formatType);\n  }\n  throw new Error('file_required');\n};\n\n// Giphy\nexport const getGiphyTrends = (context: ApiContext, limit: number = 30): Promise<any[]> => {\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/trending?api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return httpGet(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getGiphySearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> => {\n  const formatQuery: string = encodeURI(query);\n  const gifUrl: string = `http://api.giphy.com/v1/gifs/search?q=${formatQuery}&api_key=${Config.get('giphy.key')}&limit=${limit}`;\n\n  return fetch(gifUrl)\n    .then((res: Response) => res.json())\n    .then((json) => json.data.map((gifImage = {id: null, images: null}) => {\n      const {\n        id,\n        images: {\n          original: {url = ''} = {},\n          fixed_height_small: {url: thumb = ''} = {}\n        } = {}\n      } = gifImage;\n\n      return {\n        id,\n        thumb,\n        type: 'giphy',\n        url\n      };\n    }));\n};\n\nexport const getYouTubeTrends = (context: ApiContext, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.videos.list({\n      chart: 'mostPopular',\n      maxResults: limit,\n      part: ['snippet'],\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const list = data.items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n\nexport const getYouTubeSearch = (context: ApiContext, query: string, limit: number = 30): Promise<any[]> =>\n  new Promise((resolve, reject) => {\n    youtube.search.list({\n      maxResults: limit,\n      part: ['snippet'],\n      // eslint-disable-next-line\n      q: query,\n      regionCode: 'US'\n    }, (error: Error, data: any) => {\n      if(error) {\n        console.error(error);\n        reject(new Error(error[0].message));\n      } else if(data) {\n        const {items} = data;\n        const list = items.map((item) => ({\n          id: item.id,\n          thumb: item.snippet.thumbnails.high.url,\n          type: 'youtube',\n          url: `http://www.youtube.com/embed/${item.id}`\n        }));\n\n        resolve(list);\n      }\n    });\n  });\n\n// Files\nexport const getPathUserFiles = (userId: string, filename: string): string => `users/${userId}/files/${filename}`;\n\nexport const getUrlUserFiles = (userId: string, filename: string, dir: string = 'files', type: string = 'profile'): string => {\n  if(filename) {\n    return `https://box.${Config.get('app.url')}/users/${userId}/${dir}/${filename}`;\n  }\n\n  if(type === 'profile') {\n    return `https://box.${Config.get('app.url')}/defaults/user_bk.jpg`;\n  }\n\n  return `https://box.${Config.get('app.url')}/defaults/user_wh.jpg`;\n};\n\nexport const createFile = (db: Database, file: FileType): Promise<FileType> => {\n  const {fileId} = file;\n  const insert: any = {\n    _key: fileId,\n    added: Date.now()\n  };\n\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${fileId}}\n    INSERT ${insert}\n    UPDATE {}\n    IN files RETURN NEW`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((updatedFile: FileType = {}) => updatedFile)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const linkFiles = (db: Database, files: FileType[], postId: string): Promise<any> =>\n  Promise.all(\n    files.map((file: FileType) => createFile(db, file)\n      .then((file: FileType) => createPostEdge(db, file, postId)))\n  );\n\nexport const updateFiles = (db: Database, postId: string, files: FileType[]): Promise<any> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n\n  return edgeCollection.inEdges(postId)\n    .then((edges: any) => {\n      if(edges.length) {\n        // Remove linked edges\n        return Promise.all(\n          edges.map((edge) => {\n            const {_key: edgeKey} = edge;\n            const aqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN isPosted`;\n\n            return db.query(aqlQry).catch((error: Error) => {\n              throw error;\n            });\n          }))\n          .then(() => {\n            if(files.length) {\n              // Link files\n              return linkFiles(db, files, postId).then(() => files);\n            }\n            return files;\n          });\n      } else if(files.length) {\n        // Link files\n        return linkFiles(db, files, postId).then(() => files);\n      }\n      return files;\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const encodeBase64 = (buffer: Buffer): string => Buffer.from(buffer).toString('base64');\n\nexport const decodeBase64 = (dataString: string): object => {\n  const getData = (str: string) => str.match(/^data:([A-Za-z-+\\/]+);base64,(.+)$/) || [];\n  // const getData = (base64: string) => base64.substr(base64.indexOf(',') + 1);\n  let matches = getData(dataString);\n\n  if(matches.length !== 3) {\n    // If invalid make sure we don't need to decode\n    matches = getData(decodeURIComponent(dataString));\n\n    // Check it again.\n    if(matches.length !== 3) {\n      throw Error('Invalid input string');\n    }\n  }\n\n  return {\n    data: Buffer.from(matches[2], 'base64'),\n    type: matches[1]\n  };\n};\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAIA,wBAA6B;AAC7B,mBAA+C;AAC/C,sBAA4B;AAI5B,wBAAqB;AACrB,4BAAwB;AACxB,qBAAoB;AACpB,cAAyB;AAEzB,oBAAqB;AAGrB,oBAA8B;AAC9B,mBAA6B;AAE7B,MAAM,UAAU,yBAAO,QAAQ,EAAC,MAAM,qBAAO,IAAI,eAAe,SAAS;AACzE,QAAQ,SAAS,EAAC,UAAU;AAKrB,MAAM,UAAU,CAAC,SAAqB,OAAiB,OAA0B;AACtF,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,WAAW,iBAAe;AAC7D,QAAM;AAAA,IACJ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,MACE;AAGJ,QAAM,eAAuB,SAAS,0BAAQ,UAAU,6BAAW,QAAQ;AAG3E,QAAM,QAAiB,4BAAQ;AAG/B,MAAI,aAAqB,8BAAY,MAAM;AAC3C,MAAI,aAAqB,8BAAY,UAAU;AAE/C,MAAG,eAAe,MAAM,OAAO;AAC7B,iBAAa,IAAI,UAAU,IAAI,YAAY,OAAO;AAAA;AAGpD,MAAG,eAAe,IAAI;AACpB,UAAM,UAAoB,WAAW,MAAM;AAC3C,UAAM,MAAc,QAAQ,QAAQ,SAAS;AAE7C,YAAO;AAAA,WACA;AAAA,WACA;AACH,qBAAa;AACb;AAAA,WACG;AACH,qBAAa;AACb;AAAA,WACG;AACH,qBAAa;AACb;AAAA;AAEA;AAAA;AAAA;AAIN,MAAI;AAEJ,UAAO;AAAA,SACA;AAAA,SACA;AACH,gBAAU;AACV;AAAA;AAEA,gBAAU;AACV;AAAA;AAIJ,QAAM,aAAqB,8BAAY,aAAa;AAGpD,MAAG,CAAC,WAAW,eAAe,GAAG;AAC/B,UAAM,IAAI,gCAAU;AAAA;AAGtB,QAAM,WAAW,CAAC,WAAqB;AACrC,UAAM,SAAmB,6BAAa;AAEtC,WAAO,SAAS,MAAM,QACnB,KAAK,CAAC,WAAwB,OAAO,QACrC,KAAK,CAAC,OAAO,OAAO,MACpB,MAAM,CAAC,UAAiB;AACvB,YAAM;AAAA;AAAA;AAIZ,QAAM,aAAa,CAAC,KAAa,eAAuB;AACtD,UAAM,MAAc,KAAK;AAGzB,QAAG,SAAS;AACV,aAAO,mCAAgB,SAAS,cAAc,KAAK,YAChD,KAAK,CAAC,iBAA2B;AAChC,cAAM,UAAmB,iCACpB,eADoB;AAAA,UAEvB,MAAM;AAAA,UACN,OAAO;AAAA,UACP,aAAa;AAAA,UACb,UAAU;AAAA,UACV,UAAU;AAAA,UACV,MAAM;AAAA,UACN,QAAQ;AAAA;AAGV,eAAO,SAAS;AAAA,SAEjB,MAAM,CAAC,UAAiB;AACvB,cAAM;AAAA;AAAA;AAGZ,UAAM,SAAmB;AAAA,MACvB,MAAM;AAAA,MACN,OAAO;AAAA,MACP,aAAa;AAAA,MACb,UAAU;AAAA,MACV,UAAU;AAAA,MACV,MAAM;AAAA,MACN,QAAQ;AAAA;AAGV,WAAO,SAAS;AAAA;AAIlB,MAAG,OAAO;AACR,WAAO,QAAQ,IAAI,EAAC,UAAU,MAAM,KAAK,OACtC,KAAK,CAAC,SAAS,WAAW,OAAO,KAAK,MAAM,WAAW,aACvD,MAAM,MAAM;AACX,YAAM,IAAI,gCAAU;AAAA;AAAA,aAEhB,KAAK,WAAW,IAAI;AAC5B,UAAM,SAAiB,OAAO,KAAK,KAAK;AACxC,WAAO,WAAW,QAAQ;AAAA;AAE5B,QAAM,IAAI,MAAM;AAAA;AAIX,MAAM,iBAAiB,CAAC,SAAqB,QAAgB,OAAuB;AACzF,QAAM,SAAiB,iDAAiD,qBAAO,IAAI,sBAAsB;AAEzG,SAAO,2BAAQ,QACZ,KAAK,CAAC,QAAkB,IAAI,QAC5B,KAAK,CAAC,SAAS,KAAK,KAAK,IAAI,CAAC,WAAW,EAAC,IAAI,MAAM,QAAQ,WAAU;AACrE,UAAM;AAAA,MACJ;AAAA,MACA,QAAQ;AAAA,QACN,UAAU,EAAC,MAAM,OAAM;AAAA,QACvB,oBAAoB,EAAC,KAAK,QAAQ,OAAM;AAAA,UACtC;AAAA,QACF;AAEJ,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA,MAAM;AAAA,MACN;AAAA;AAAA;AAAA;AAKD,MAAM,iBAAiB,CAAC,SAAqB,OAAe,QAAgB,OAAuB;AACxG,QAAM,cAAsB,UAAU;AACtC,QAAM,SAAiB,yCAAyC,uBAAuB,qBAAO,IAAI,sBAAsB;AAExH,SAAO,MAAM,QACV,KAAK,CAAC,QAAkB,IAAI,QAC5B,KAAK,CAAC,SAAS,KAAK,KAAK,IAAI,CAAC,WAAW,EAAC,IAAI,MAAM,QAAQ,WAAU;AACrE,UAAM;AAAA,MACJ;AAAA,MACA,QAAQ;AAAA,QACN,UAAU,EAAC,MAAM,OAAM;AAAA,QACvB,oBAAoB,EAAC,KAAK,QAAQ,OAAM;AAAA,UACtC;AAAA,QACF;AAEJ,WAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA,MAAM;AAAA,MACN;AAAA;AAAA;AAAA;AAKD,MAAM,mBAAmB,CAAC,SAAqB,QAAgB,OACpE,IAAI,QAAQ,CAAC,SAAS,WAAW;AAC/B,UAAQ,OAAO,KAAK;AAAA,IAClB,OAAO;AAAA,IACP,YAAY;AAAA,IACZ,MAAM,CAAC;AAAA,IACP,YAAY;AAAA,KACX,CAAC,OAAc,SAAc;AAC9B,QAAG,OAAO;AACR,cAAQ,MAAM;AACd,aAAO,IAAI,MAAM,MAAM,GAAG;AAAA,eAClB,MAAM;AACd,YAAM,OAAO,KAAK,MAAM,IAAI,CAAC,SAAU;AAAA,QACrC,IAAI,KAAK;AAAA,QACT,OAAO,KAAK,QAAQ,WAAW,KAAK;AAAA,QACpC,MAAM;AAAA,QACN,KAAK,gCAAgC,KAAK;AAAA;AAG5C,cAAQ;AAAA;AAAA;AAAA;AAMT,MAAM,mBAAmB,CAAC,SAAqB,OAAe,QAAgB,OACnF,IAAI,QAAQ,CAAC,SAAS,WAAW;AAC/B,UAAQ,OAAO,KAAK;AAAA,IAClB,YAAY;AAAA,IACZ,MAAM,CAAC;AAAA,IAEP,GAAG;AAAA,IACH,YAAY;AAAA,KACX,CAAC,OAAc,SAAc;AAC9B,QAAG,OAAO;AACR,cAAQ,MAAM;AACd,aAAO,IAAI,MAAM,MAAM,GAAG;AAAA,eAClB,MAAM;AACd,YAAM,EAAC,UAAS;AAChB,YAAM,OAAO,MAAM,IAAI,CAAC,SAAU;AAAA,QAChC,IAAI,KAAK;AAAA,QACT,OAAO,KAAK,QAAQ,WAAW,KAAK;AAAA,QACpC,MAAM;AAAA,QACN,KAAK,gCAAgC,KAAK;AAAA;AAG5C,cAAQ;AAAA;AAAA;AAAA;AAMT,MAAM,mBAAmB,CAAC,QAAgB,aAA6B,SAAS,gBAAgB;AAEhG,MAAM,kBAAkB,CAAC,QAAgB,UAAkB,MAAc,SAAS,OAAe,cAAsB;AAC5H,MAAG,UAAU;AACX,WAAO,eAAe,qBAAO,IAAI,oBAAoB,UAAU,OAAO;AAAA;AAGxE,MAAG,SAAS,WAAW;AACrB,WAAO,eAAe,qBAAO,IAAI;AAAA;AAGnC,SAAO,eAAe,qBAAO,IAAI;AAAA;AAG5B,MAAM,aAAa,CAAC,IAAc,SAAsC;AAC7E,QAAM,EAAC,WAAU;AACjB,QAAM,SAAc;AAAA,IAClB,MAAM;AAAA,IACN,OAAO,KAAK;AAAA;AAGd,QAAM,SAAmB,oCAAoB;AAAA,aAClC;AAAA;AAAA;AAIX,SAAO,GAAG,MAAM,QACb,KAAK,CAAC,WAAwB,OAAO,QACrC,KAAK,CAAC,cAAwB,OAAO,aACrC,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;AAIL,MAAM,YAAY,CAAC,IAAc,OAAmB,WACzD,QAAQ,IACN,MAAM,IAAI,CAAC,SAAmB,WAAW,IAAI,MAC1C,KAAK,CAAC,UAAmB,iCAAe,IAAI,OAAM;AAGlD,MAAM,cAAc,CAAC,IAAc,QAAgB,UAAoC;AAC5F,QAAM,iBAAiC,GAAG,WAAW;AAErD,SAAO,eAAe,QAAQ,QAC3B,KAAK,CAAC,UAAe;AACpB,QAAG,MAAM,QAAQ;AAEf,aAAO,QAAQ,IACb,MAAM,IAAI,CAAC,SAAS;AAClB,cAAM,EAAC,MAAM,YAAW;AACxB,cAAM,SAAmB,mCAAmB;AAE5C,eAAO,GAAG,MAAM,QAAQ,MAAM,CAAC,UAAiB;AAC9C,gBAAM;AAAA;AAAA,UAGT,KAAK,MAAM;AACV,YAAG,MAAM,QAAQ;AAEf,iBAAO,UAAU,IAAI,OAAO,QAAQ,KAAK,MAAM;AAAA;AAEjD,eAAO;AAAA;AAAA,eAEH,MAAM,QAAQ;AAEtB,aAAO,UAAU,IAAI,OAAO,QAAQ,KAAK,MAAM;AAAA;AAEjD,WAAO;AAAA,KAER,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA;AAAA;AAIL,MAAM,eAAe,CAAC,WAA2B,OAAO,KAAK,QAAQ,SAAS;AAE9E,MAAM,eAAe,CAAC,eAA+B;AAC1D,QAAM,UAAU,CAAC,QAAgB,IAAI,MAAM,yCAAyC;AAEpF,MAAI,UAAU,QAAQ;AAEtB,MAAG,QAAQ,WAAW,GAAG;AAEvB,cAAU,QAAQ,mBAAmB;AAGrC,QAAG,QAAQ,WAAW,GAAG;AACvB,YAAM,MAAM;AAAA;AAAA;AAIhB,SAAO;AAAA,IACL,MAAM,OAAO,KAAK,QAAQ,IAAI;AAAA,IAC9B,MAAM,QAAQ;AAAA;AAAA;",
  "names": []
}

|