Press n or j to go to the next uncovered block, b, p or k for the previous block.
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 | import { LambdaApiFunction, handleResourceApi } from './_base'; import { api401Body, generateApi400Body } from '@/types/api/_shared'; import { FullFileObject } from '@/types/api/files'; import { GetPagesApi, getPagesApiQueryValidator } from '@/types/api/pages'; import { TypedQueryInput } from '@/types/backend/dynamo'; import { TABLE_FILE, typedQuery } from '@/utils/backend/dynamoTyped'; import { validateObject } from '@/utils/backend/validation'; import { getLogger } from '@/utils/common/logger'; const logger = getLogger('resources/api/v2/pages'); const defaultListLimit = 100; const GET: LambdaApiFunction<GetPagesApi> = async function (event, user, userPerms) { logger.trace('GET', ...arguments); // Validate the user if ( user === null || !userPerms.isUser ) { return [ 401, api401Body, ]; } // Validate the query const [ query, queryErrors, ] = validateObject<GetPagesApi['query']>( event.multiValueQueryStringParameters || {}, getPagesApiQueryValidator, true ); if ( query === null || queryErrors.length > 0 ) { return [ 400, generateApi400Body(queryErrors), ]; } const queryConfig: TypedQueryInput<FullFileObject> & Required<Pick< TypedQueryInput<FullFileObject>, 'ExpressionAttributeNames' | 'ExpressionAttributeValues' | 'KeyConditionExpression' >> = { ScanIndexForward: false, TableName: TABLE_FILE, Limit: defaultListLimit, IndexName: 'ToneIndex', ExpressionAttributeNames: { '#ToneIndex': 'ToneIndex', }, ExpressionAttributeValues: { ':ToneIndex': 'y', }, KeyConditionExpression: '#ToneIndex = :ToneIndex', }; if (typeof query.tg !== 'undefined') { queryConfig.ExpressionAttributeNames['#Talkgroup'] = 'Talkgroup'; queryConfig.FilterExpression = query.tg.map((tg, idx) => { queryConfig.ExpressionAttributeValues[`:Talkgroup${idx}`] = tg; return `#Talkgroup = :Talkgroup${idx}`; }).join(' OR '); } if (typeof query.before !== 'undefined') { queryConfig.ExpressionAttributeNames['#StartTime'] = 'StartTime'; queryConfig.ExpressionAttributeValues[':StartTime'] = query.before; queryConfig.KeyConditionExpression += ' AND #StartTime < :StartTime'; } const queryResults = await typedQuery<FullFileObject>(queryConfig); if (!queryResults.Items?.length) { return [ 200, { before: null, files: [], query: queryResults, }, ]; } let before: number | null = null; queryResults.Items.forEach(f => { if ( f.StartTime && ( before === null || f.StartTime < before ) ) { before = f.StartTime; } }); return [ 200, { before, files: queryResults.Items, query: queryResults, }, ]; }; export const main = handleResourceApi.bind(null, { GET, }); |