From 907a85690aaba3f66d7f6c748d8b2719e04f391b Mon Sep 17 00:00:00 2001 From: Wolmin <44748271+Wolmin@users.noreply.github.com> Date: Mon, 14 Oct 2024 10:10:47 +0200 Subject: [PATCH] feat: Algolia Removal (#23) * Initial graph client impl * Fetching logic * Generalize types * Validate graph env * Trim name Signed-off-by: Wolmin --- deploy/tools/envs-validator/schema.ts | 8 ++--- docs/ENVS.md | 6 ++-- lib/api/buildUniversalProfileUrl.ts | 11 ------ lib/api/graphClient.ts | 47 +++++++++++++++++++++++++ lib/api/graphQueries.ts | 25 +++++++++++++ lib/api/graphTypes.ts | 18 ++++++++++ lib/api/useUniversalProfileApiFetch.ts | 20 +++++++---- nextjs/csp/policies/universalprofile.ts | 1 + types/api/universalProfile.ts | 8 +++++ 9 files changed, 116 insertions(+), 28 deletions(-) delete mode 100644 lib/api/buildUniversalProfileUrl.ts create mode 100644 lib/api/graphClient.ts create mode 100644 lib/api/graphQueries.ts create mode 100644 lib/api/graphTypes.ts diff --git a/deploy/tools/envs-validator/schema.ts b/deploy/tools/envs-validator/schema.ts index 0c69c0413e..530220597f 100644 --- a/deploy/tools/envs-validator/schema.ts +++ b/deploy/tools/envs-validator/schema.ts @@ -909,17 +909,13 @@ const schema = yup NEXT_PUBLIC_RE_CAPTCHA_V3_APP_SITE_KEY: yup.string(), // DEPRECATED NEXT_PUBLIC_GOOGLE_ANALYTICS_PROPERTY_ID: yup.string(), NEXT_PUBLIC_MIXPANEL_PROJECT_TOKEN: yup.string(), + NEXT_PUBLIC_UNIVERSAL_PROFILES_API_URL: yup.string(), + NEXT_PUBLIC_GRAPH_URL: yup.string(), NEXT_PUBLIC_GROWTH_BOOK_CLIENT_KEY: yup.string(), NEXT_PUBLIC_ROLLBAR_CLIENT_TOKEN: yup.string(), - NEXT_PUBLIC_FAVICON_GENERATOR_API_KEY: yup.string(), - NEXT_PUBLIC_UP_API_URL: yup.string(), - NEXT_PUBLIC_UNIVERSAL_PROFILES_API_URL: yup.string(), // Misc NEXT_PUBLIC_USE_NEXT_JS_PROXY: yup.boolean(), - NEXT_PUBLIC_ALGOLIA_APP_ID: yup.string(), - NEXT_PUBLIC_ALGOLIA_API_KEY: yup.string(), - NEXT_PUBLIC_ALGOLIA_INDEX_NAME: yup.string(), }) .concat(accountSchema) .concat(adsBannerSchema) diff --git a/docs/ENVS.md b/docs/ENVS.md index 62207dd60a..b51f9976c1 100644 --- a/docs/ENVS.md +++ b/docs/ENVS.md @@ -257,10 +257,8 @@ Settings for meta tags, OG tags and SEO | NEXT_PUBLIC_VIEWS_CONTRACT_SOLIDITYSCAN_ENABLED | `boolean` | Set to `true` if SolidityScan reports are supported | - | - | `true` | v1.19.0+ | | NEXT_PUBLIC_VIEWS_CONTRACT_EXTRA_VERIFICATION_METHODS | `Array<'solidity-hardhat' \| 'solidity-foundry'>` | Pass an array of additional methods from which users can choose while verifying a smart contract. Both methods are available by default, pass `'none'` string to disable them all. | - | - | `['solidity-hardhat']` | v1.33.0+ | | NEXT_PUBLIC_VIEWS_CONTRACT_LANGUAGE_FILTERS | `Array<'solidity' \| 'vyper' \| 'yul' \| 'scilla'>` | Pass an array of contract languages that will be displayed as options in the filter on the verified contract page. | - | `['solidity','vyper','yul']` | `['solidity','vyper','yul','scilla']` | v1.37.0+ | -| NEXT_PUBLIC_UNIVERSAL_PROFILES_API_URL | `string` | LUKSO UP API URL used for getting various UP data | - | - | `https://api.universalprofile.com` | -| NEXT_PUBLIC_ALGOLIA_APP_ID | `string` | Algolia App's ID | - | - | `ABCD1234` | -| NEXT_PUBLIC_ALGOLIA_API_KEY | `string` | Algolia API key used for authenticating requests | - | - | `abcd1234defg5678` | -| NEXT_PUBLIC_ALGOLIA_INDEX_NAME | `string` | Index name from where data should be requested | - | - | `prod_mainnet_data` | +| NEXT_PUBLIC_UNIVERSAL_PROFILES_API_URL | `string` | LUKSO UNIVERSAL PROFILE API URL used for getting various universal profile data | - | - | `https://api.universalprofile.com` | - | +| NEXT_PUBLIC_GRAPH_URL | `string` | URL of LUKSO Graph Indexer | - | - | `https://envio.mainnet.lukso.dev/`| - | ##### Address views list | Id | Description | diff --git a/lib/api/buildUniversalProfileUrl.ts b/lib/api/buildUniversalProfileUrl.ts deleted file mode 100644 index 690da58e06..0000000000 --- a/lib/api/buildUniversalProfileUrl.ts +++ /dev/null @@ -1,11 +0,0 @@ -import algoliasearch from 'algoliasearch'; - -import { getEnvValue } from '../../configs/app/utils'; - -const algolia = { - appId: getEnvValue('NEXT_PUBLIC_ALGOLIA_APP_ID') || '', - apiKey: getEnvValue('NEXT_PUBLIC_ALGOLIA_API_KEY') || '', - index: getEnvValue('NEXT_PUBLIC_ALGOLIA_INDEX_NAME') || '', -}; - -export const algoliaIndex = algoliasearch(algolia.appId, algolia.apiKey).initIndex(algolia.index); diff --git a/lib/api/graphClient.ts b/lib/api/graphClient.ts new file mode 100644 index 0000000000..01429d2784 --- /dev/null +++ b/lib/api/graphClient.ts @@ -0,0 +1,47 @@ +import type { GraphResponse } from 'types/api/universalProfile'; + +import { getEnvValue } from 'configs/app/utils'; + +import { constructQuery } from './graphQueries'; +import type { QueryOperation, SearchProfileQueryResponse } from './graphTypes'; + +const graphUrl = getEnvValue('NEXT_PUBLIC_GRAPH_URL') || ''; + +type FetchFunc = (queryParams?: string) => Promise | null>; + +type GraphClient = { + getProfiles: FetchFunc; +}; + +const queryParamsToGraphQuery = (operationName: QueryOperation, queryParams?: string): string => { + const query = constructQuery(operationName, queryParams); + + return JSON.stringify({ + operationName: operationName, + query: query, + }); +}; + +const fetchQuery = (operationName: QueryOperation): FetchFunc => { + return async(queryParams?: string): Promise | null> => { + const query = queryParamsToGraphQuery(operationName, queryParams); + + try { + const resp = await fetch(graphUrl, { + method: 'POST', + headers: {}, + body: query, + }); + const json = await resp.json(); + return json as GraphResponse; + } catch (err) { + return null; + } + }; +}; + +const createGraphClient = (): GraphClient => ({ + getProfiles: fetchQuery('search_profiles'), +}); + +export const graphClient: GraphClient = createGraphClient(); diff --git a/lib/api/graphQueries.ts b/lib/api/graphQueries.ts new file mode 100644 index 0000000000..90d36ce51f --- /dev/null +++ b/lib/api/graphQueries.ts @@ -0,0 +1,25 @@ +import type { QueryOperation } from './graphTypes'; + +type QueryConstructor = (queryParams?: string) => string; + +export const constructQuery = (operationType: QueryOperation, queryParams?: string) => { + const queryConstruct = queryConstructors[operationType]; + + return queryConstruct(queryParams); +}; + +export const searchProfileQuery = (queryParams?: string): string => `query search_profiles { + search_profiles(args: { search: "${ queryParams }" }) { + profileImages(order_by: { width: asc }) { + src, + width, + }, + id, + name, + fullName, + } +}`; + +const queryConstructors: Record = { + search_profiles: searchProfileQuery, +}; diff --git a/lib/api/graphTypes.ts b/lib/api/graphTypes.ts new file mode 100644 index 0000000000..cd9f95430b --- /dev/null +++ b/lib/api/graphTypes.ts @@ -0,0 +1,18 @@ +export type ProfileImage = { + src: string; + width: number; +}; + +export type SearchProfileQueryResponse = { + profileImages: Array; + id: string; + name: string; + fullName: string; +}; + +export type GraphQuery = { + operationName: string; + query: string; +}; + +export type QueryOperation = 'search_profiles'; diff --git a/lib/api/useUniversalProfileApiFetch.ts b/lib/api/useUniversalProfileApiFetch.ts index 9eeea0a0b3..aaa71d7a12 100644 --- a/lib/api/useUniversalProfileApiFetch.ts +++ b/lib/api/useUniversalProfileApiFetch.ts @@ -1,11 +1,11 @@ import React from 'react'; import type { SearchResultAddressOrContractOrUniversalProfile } from '../../types/api/search'; -import type { UniversalProfileProxyResponse } from '../../types/api/universalProfile'; import type { Params as FetchParams } from 'lib/hooks/useFetch'; -import { algoliaIndex } from './buildUniversalProfileUrl'; +import { graphClient } from './graphClient'; +import type { SearchProfileQueryResponse } from './graphTypes'; import { isUniversalProfileEnabled } from './isUniversalProfileEnabled'; import type { ResourceName, ResourcePathParams } from './resources'; @@ -22,13 +22,19 @@ export default function useUniversalProfileApiFetch() { return [] as Array; } try { - const { hits } = await algoliaIndex.search(queryParams); - return hits.map((hit) => { - const hitAsUp = hit as unknown as UniversalProfileProxyResponse; + const result = await graphClient.getProfiles(queryParams); + if (result == null) { + return [] as Array; + } + + const hits = result.data.search_profiles as Array; + + return hits.map((hit: SearchProfileQueryResponse) => { + const hitAsUp = hit as unknown as SearchProfileQueryResponse; return { type: 'universal_profile', - name: hitAsUp.hasProfileName ? hitAsUp.LSP3Profile.name : null, - address: hit.objectID, + name: hitAsUp.name !== '' ? hitAsUp.name.trim() : null, + address: hit.id, is_smart_contract_verified: false, }; }); diff --git a/nextjs/csp/policies/universalprofile.ts b/nextjs/csp/policies/universalprofile.ts index 886889d327..14d516913f 100644 --- a/nextjs/csp/policies/universalprofile.ts +++ b/nextjs/csp/policies/universalprofile.ts @@ -6,6 +6,7 @@ export function universalProfile(): CspDev.DirectiveDescriptor { 'api.universalprofile.cloud', '*.algolianet.com', '*.algolia.net', + 'envio.mainnet.lukso.dev', ], }; } diff --git a/types/api/universalProfile.ts b/types/api/universalProfile.ts index 3fa7241c48..5051b980c8 100644 --- a/types/api/universalProfile.ts +++ b/types/api/universalProfile.ts @@ -1,3 +1,5 @@ +import type { QueryOperation } from 'lib/api/graphTypes'; + export type UniversalProfileProxyResponse = { type: string; hasProfileName: boolean; @@ -25,3 +27,9 @@ export type UniversalProfileAlgoliaResponse = { }; }; }; + +export type GraphResponse = { + data: { + [key in QueryOperation]: Array; + }; +};