From 311a6874a4c2170bf2eacf64801aa0b12641a6a6 Mon Sep 17 00:00:00 2001 From: Joel Carter Date: Thu, 5 Oct 2023 10:25:52 -0500 Subject: [PATCH 1/6] Query on status or success --- .../image-thumbnail/src/Diviner/Query.ts | 2 +- .../src/Payload/ImageThumbnailResult.ts | 3 ++- .../packages/diviner/src/Diviner/Diviner.ts | 18 ++++++++++-------- 3 files changed, 13 insertions(+), 10 deletions(-) diff --git a/packages/plugins/packages/payload/packages/image-thumbnail/src/Diviner/Query.ts b/packages/plugins/packages/payload/packages/image-thumbnail/src/Diviner/Query.ts index 7a053ae9841..4a912e517d9 100644 --- a/packages/plugins/packages/payload/packages/image-thumbnail/src/Diviner/Query.ts +++ b/packages/plugins/packages/payload/packages/image-thumbnail/src/Diviner/Query.ts @@ -7,6 +7,6 @@ export type ImageThumbnailDivinerQuerySchema = `${ImageThumbnailDivinerSchema}.q export const ImageThumbnailDivinerQuerySchema: ImageThumbnailDivinerQuerySchema = `${ImageThumbnailDivinerSchema}.query` export type ImageThumbnailDivinerQuery = Pick & - Payload<{ status?: boolean; url: string }, ImageThumbnailDivinerQuerySchema> + Payload<{ status?: number; success?: boolean; url: string }, ImageThumbnailDivinerQuerySchema> export const isImageThumbnailDivinerQuery = isPayloadOfSchemaType(ImageThumbnailDivinerQuerySchema) diff --git a/packages/plugins/packages/payload/packages/image-thumbnail/src/Payload/ImageThumbnailResult.ts b/packages/plugins/packages/payload/packages/image-thumbnail/src/Payload/ImageThumbnailResult.ts index c54c5dc5f78..aacb9565f37 100644 --- a/packages/plugins/packages/payload/packages/image-thumbnail/src/Payload/ImageThumbnailResult.ts +++ b/packages/plugins/packages/payload/packages/image-thumbnail/src/Payload/ImageThumbnailResult.ts @@ -7,7 +7,8 @@ export type ImageThumbnailResultIndexSchema = typeof ImageThumbnailResultIndexSc export interface ImageThumbnailResultInfo { sources: string[] - status: boolean + status?: number + success: boolean timestamp: number url: string } diff --git a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts index ae07c2d920d..a02037be255 100644 --- a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts +++ b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts @@ -16,6 +16,7 @@ import { ImageThumbnailDivinerParams, ImageThumbnailResult, ImageThumbnailResultIndexSchema, + ImageThumbnailResultInfo, ImageThumbnailSchema, isImageThumbnail, isImageThumbnailDivinerQuery, @@ -37,7 +38,7 @@ type ConfigStore = Extract /** * The fields that will need to be indexed on in the underlying store */ -type QueryableImageThumbnailResultProperties = Extract +type QueryableImageThumbnailResultProperties = Extract /** * The query that will be used to retrieve the results from the underlying store @@ -129,11 +130,11 @@ export class ImageThumbnailDiviner { const { sourceUrl: url } = thumbnailPayload const { timestamp } = timestampPayload - const status = thumbnailPayload.http?.status ? true : false + const status = thumbnailPayload.http?.status + const success = thumbnailPayload.http?.status ? true : false const sources = [boundWitnessHash, thumbnailHash, timestampHash] - const result = new PayloadBuilder({ schema: ImageThumbnailResultIndexSchema }) - .fields({ sources, status, timestamp, url }) - .build() + const fields = status ? { sources, status, success, timestamp, url } : { sources, success, timestamp, url } + const result = new PayloadBuilder({ schema: ImageThumbnailResultIndexSchema }).fields(fields).build() return result }, ) @@ -168,13 +169,14 @@ export class ImageThumbnailDiviner { - const { limit: payloadLimit, offset: payloadOffset, order: payloadOrder, status: payloadStatus, url } = payload + const { limit: payloadLimit, offset: payloadOffset, order: payloadOrder, status: payloadStatus, success: payloadSuccess, url } = payload const limit = payloadLimit ?? 1 const order = payloadOrder ?? 'desc' const offset = payloadOffset ?? 0 - const status = payloadStatus ?? true + const status = payloadStatus ?? 200 + const success = payloadSuccess ?? true const query = new PayloadBuilder({ schema: PayloadDivinerQuerySchema }) - .fields({ limit, offset, order, status, url }) + .fields({ limit, offset, order, status, success, url }) .build() return await diviner.divine([query]) }), From fb4020b0f72ebecdd7451d4f158ff4eb691d3374 Mon Sep 17 00:00:00 2001 From: Joel Carter Date: Thu, 5 Oct 2023 10:37:52 -0500 Subject: [PATCH 2/6] Add tests for filter criteria --- .../diviner/src/Diviner/spec/Diviner.spec.ts | 82 +++++++++++++------ 1 file changed, 59 insertions(+), 23 deletions(-) diff --git a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts index 082f98c538d..4f6cfbc23b6 100644 --- a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts +++ b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts @@ -15,27 +15,32 @@ describe('ImageThumbnailDiviner', () => { const indexArchivistName = 'indexArchivist' const stateArchivistName = 'stateArchivist' const thumbnailArchivistName = 'thumbnailArchivist' - - const successfulThumbnail = { - _timestamp: 1692812290173, + const sourceUrl = 'https://placekitten.com/200/300' + const thumbnailHttpSuccess = { http: { status: 200, }, schema: 'network.xyo.image.thumbnail', sourceHash: '7f39363514d9d9b958a5a993edeba35cb44f912c7072ed9ddd628728ac0fd681', - sourceUrl: 'https://assets.otherside.xyz/kodas/pfp_full/1200px/3724.webp', - url: '', + sourceUrl, + url: '', } - const failedThumbnail = { - _hash: 'e6696f95dfb0df7d9f5054d93d2a5d2bb77ed2763d3b8b2581bcde448d9d7757', - _timestamp: 1692812925775, + const thumbnailHttpFail = { http: { ipAddress: '104.17.96.13', status: 429, }, schema: 'network.xyo.image.thumbnail', - sourceUrl: 'https://cloudflare-ipfs.com/ipfs/QmbFBUR8y9Xhwv7nQdiy1Xafyxu3i8BaFThts2UyxM73Ge', + sourceUrl, + } + + const thumbnailWitnessFail = { + http: { + ipAddress: '104.17.96.13', + }, + schema: 'network.xyo.image.thumbnail', + sourceUrl, } let sut: ImageThumbnailDiviner @@ -69,11 +74,24 @@ describe('ImageThumbnailDiviner', () => { }) // Insert previously witnessed payloads into thumbnail archivist - const successfulTimestamp: TimeStamp = { schema: TimestampSchema, timestamp: Date.now() } - const [successfulBoundWitness, successfulPayloads] = await new BoundWitnessBuilder().payloads([successfulThumbnail, successfulTimestamp]).build() - const failedTimestamp: TimeStamp = { schema: TimestampSchema, timestamp: Date.now() } - const [failedBoundWitness, failedPayloads] = await new BoundWitnessBuilder().payloads([failedThumbnail, failedTimestamp]).build() - await thumbnailArchivist.insert([successfulBoundWitness, ...successfulPayloads, failedBoundWitness, ...failedPayloads]) + const httpSuccessTimestamp: TimeStamp = { schema: TimestampSchema, timestamp: Date.now() } + const [httpSuccessBoundWitness, httpSuccessPayloads] = await new BoundWitnessBuilder() + .payloads([thumbnailHttpSuccess, httpSuccessTimestamp]) + .build() + const httpFailTimestamp: TimeStamp = { schema: TimestampSchema, timestamp: Date.now() } + const [httpFailBoundWitness, httpFailPayloads] = await new BoundWitnessBuilder().payloads([thumbnailHttpFail, httpFailTimestamp]).build() + const witnessFailTimestamp: TimeStamp = { schema: TimestampSchema, timestamp: Date.now() } + const [witnessFailBoundWitness, witnessFailPayloads] = await new BoundWitnessBuilder() + .payloads([thumbnailWitnessFail, witnessFailTimestamp]) + .build() + await thumbnailArchivist.insert([ + httpSuccessBoundWitness, + ...httpSuccessPayloads, + httpFailBoundWitness, + ...httpFailPayloads, + witnessFailBoundWitness, + ...witnessFailPayloads, + ]) const thumbnailStore: SearchableStorage = { archivist: thumbnailArchivist.address, @@ -151,19 +169,37 @@ describe('ImageThumbnailDiviner', () => { await delay(1000) }) describe('with no thumbnail for the provided URL', () => { + const url = 'https://does.not.exist.io' + const schema = ImageThumbnailDivinerQuerySchema it('returns nothing', async () => { - const payload: ImageThumbnailDivinerQuery = { schema: ImageThumbnailDivinerQuerySchema, url: 'https://does.not.exist.io' } - const result = await sut.divine([payload]) + const query: ImageThumbnailDivinerQuery = { schema, url } + const result = await sut.divine([query]) expect(result).toBeArrayOfSize(0) }) }) - describe('with successful thumbnail for the provided URL', () => { - it('returns the most recent success', async () => { - const payload: ImageThumbnailDivinerQuery = { schema: ImageThumbnailDivinerQuerySchema, url: successfulThumbnail.sourceUrl } - const result = await sut.divine([payload]) - expect(result).toBeArrayOfSize(1) - const expected = await PayloadHasher.hashAsync(successfulThumbnail) - expect(result[0]?.sources).toContain(expected) + describe('with thumbnails for the provided URL', () => { + const url = sourceUrl + const schema = ImageThumbnailDivinerQuerySchema + describe('with no filter criteria', () => { + it('returns the most recent success', async () => { + const query: ImageThumbnailDivinerQuery = { schema, url } + const result = await sut.divine([query]) + expect(result).toBeArrayOfSize(1) + const expected = await PayloadHasher.hashAsync(thumbnailHttpSuccess) + expect(result[0]?.sources).toContain(expected) + }) + }) + describe('with filter criteria', () => { + describe('for status code', () => { + it.each([thumbnailHttpSuccess, thumbnailHttpFail])('returns the most recent instance of that status code', async (payload) => { + const { status } = payload.http + const query: ImageThumbnailDivinerQuery = { schema, status, url } + const result = await sut.divine([query]) + expect(result).toBeArrayOfSize(1) + const expected = await PayloadHasher.hashAsync(payload) + expect(result[0]?.sources).toContain(expected) + }) + }) }) }) }) From f650d3a8573aa9842b1feebe50c893c5d8f9c549 Mon Sep 17 00:00:00 2001 From: Joel Carter Date: Thu, 5 Oct 2023 11:59:07 -0500 Subject: [PATCH 3/6] Only filter on status code if success was not supplied or is true --- .../packages/diviner/src/Diviner/Diviner.ts | 10 ++++++---- .../diviner/src/Diviner/spec/Diviner.spec.ts | 14 +++++++++++++- 2 files changed, 19 insertions(+), 5 deletions(-) diff --git a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts index a02037be255..e821bdb0572 100644 --- a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts +++ b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts @@ -173,11 +173,13 @@ export class ImageThumbnailDiviner({ schema: PayloadDivinerQuerySchema }) - .fields({ limit, offset, order, status, success, url }) - .build() + const fields: Partial = { limit, offset, order, success, url } + // Only filter on status code if success was not supplied or is true + if (payloadSuccess !== undefined && payloadSuccess === true) { + fields.status = payloadStatus ?? 200 + } + const query = new PayloadBuilder({ schema: PayloadDivinerQuerySchema }).fields(fields).build() return await diviner.divine([query]) }), ) diff --git a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts index 4f6cfbc23b6..73f37d94840 100644 --- a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts +++ b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/spec/Diviner.spec.ts @@ -191,7 +191,8 @@ describe('ImageThumbnailDiviner', () => { }) describe('with filter criteria', () => { describe('for status code', () => { - it.each([thumbnailHttpSuccess, thumbnailHttpFail])('returns the most recent instance of that status code', async (payload) => { + const cases = [thumbnailHttpSuccess, thumbnailHttpFail] + it.each(cases)('returns the most recent instance of that status code', async (payload) => { const { status } = payload.http const query: ImageThumbnailDivinerQuery = { schema, status, url } const result = await sut.divine([query]) @@ -200,6 +201,17 @@ describe('ImageThumbnailDiviner', () => { expect(result[0]?.sources).toContain(expected) }) }) + describe('for success', () => { + const cases = [thumbnailHttpFail, thumbnailWitnessFail] + it.each(cases)('returns the most recent instance of that success state', async (payload) => { + const success = (payload?.http as { status?: number })?.status ? true : false + const query: ImageThumbnailDivinerQuery = { schema, success, url } + const result = await sut.divine([query]) + expect(result).toBeArrayOfSize(1) + const expected = await PayloadHasher.hashAsync(payload) + expect(result[0]?.sources).toContain(expected) + }) + }) }) }) }) From b00aea55cccbce3b8b57768705b174493cc00093 Mon Sep 17 00:00:00 2001 From: Joel Carter Date: Thu, 5 Oct 2023 12:00:08 -0500 Subject: [PATCH 4/6] Use offset if supplied --- .../payload/packages/memory/src/MemoryPayloadDiviner.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/modules/packages/diviner/packages/payload/packages/memory/src/MemoryPayloadDiviner.ts b/packages/modules/packages/diviner/packages/payload/packages/memory/src/MemoryPayloadDiviner.ts index 8eff83003e2..108037e3487 100644 --- a/packages/modules/packages/diviner/packages/payload/packages/memory/src/MemoryPayloadDiviner.ts +++ b/packages/modules/packages/diviner/packages/payload/packages/memory/src/MemoryPayloadDiviner.ts @@ -30,7 +30,7 @@ export class MemoryPayloadDiviner { const allPairs = await Promise.all( From 2feef8dd4c67e2ddd342dbc13ba241312eefe76c Mon Sep 17 00:00:00 2001 From: Joel Carter Date: Thu, 5 Oct 2023 12:04:46 -0500 Subject: [PATCH 5/6] Simplify filtering edge cases --- .../packages/diviner/src/Diviner/Diviner.ts | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts index e821bdb0572..4a7c4ab23bd 100644 --- a/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts +++ b/packages/plugins/packages/payloadset/packages/image-thumbnail/packages/diviner/src/Diviner/Diviner.ts @@ -175,10 +175,10 @@ export class ImageThumbnailDiviner = { limit, offset, order, success, url } - // Only filter on status code if success was not supplied or is true - if (payloadSuccess !== undefined && payloadSuccess === true) { - fields.status = payloadStatus ?? 200 - } + // Default to filtering on 200 status code if success was not supplied + if (payloadSuccess === undefined) fields.status = payloadStatus ?? 200 + // If success is true and status was supplied, use it + if (success === true && payloadStatus !== undefined) fields.status = payloadStatus const query = new PayloadBuilder({ schema: PayloadDivinerQuerySchema }).fields(fields).build() return await diviner.divine([query]) }), From 23348052a58ea66c3bb13df1b6aeedfad3f43380 Mon Sep 17 00:00:00 2001 From: Joel Carter Date: Thu, 5 Oct 2023 12:09:59 -0500 Subject: [PATCH 6/6] Don't assert exact collection counts --- .../witness/src/lib/spec/getNftCollectionCount.spec.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/plugins/packages/payloadset/packages/crypto/packages/nft/packages/collection/witness/src/lib/spec/getNftCollectionCount.spec.ts b/packages/plugins/packages/payloadset/packages/crypto/packages/nft/packages/collection/witness/src/lib/spec/getNftCollectionCount.spec.ts index 93240b26de6..071cb8de440 100644 --- a/packages/plugins/packages/payloadset/packages/crypto/packages/nft/packages/collection/witness/src/lib/spec/getNftCollectionCount.spec.ts +++ b/packages/plugins/packages/payloadset/packages/crypto/packages/nft/packages/collection/witness/src/lib/spec/getNftCollectionCount.spec.ts @@ -19,6 +19,6 @@ describeIf(process.env.INFURA_PROJECT_ID)('getNftCollectionCount', () => { it.each(cases)('gets NFTs owned by the address', async (address, chainId, expected) => { const result = await getNftCollectionCount(address, chainId, privateKey) expect(result).toBeNumber() - expect(result).toEqual(expected) + expect(result).toBeGreaterThanOrEqual(expected) }) })