UNPKG

@ixily/activ

Version:

Alpha Capture Trade Idea Verification. Blockchain ownership proven trade ideas and strategies.

904 lines (848 loc) 39.7 kB
import { LogModule as log } from '..' import { ProvableIdeasModule } from './provable-ideas.module' import { CONTRACT_CONSTANTS, CONTRACT_INTERFACES, CacheService, Contract, ContractModule, IContract, ethers, getBoolean, isNullOrUndefined, skipExpirationInSeconds, } from '../../' interface IIdeaByNftIdSearchParams { ticker?: string type?: 'public' | 'close' strategyKey?: string } const isCloseIdea = (idea: CONTRACT_INTERFACES.ITradeIdea): boolean => { const ICanSeeThatThisIsOneIdeaPublicAndIsACloseOne = typeof idea.idea === 'string' ? false : (idea.idea as CONTRACT_INTERFACES.ITradeIdeaIdea).kind === 'close' return ICanSeeThatThisIsOneIdeaPublicAndIsACloseOne } const getETHBalance = async (contract?: Contract): Promise<number> => { contract = ContractModule.thisOrGetGate(contract) const balance = await contract.signer.getBalance() const balanceInEth = ethers.utils.formatEther(balance) return Number(Number(balanceInEth || 0).toFixed(4)) } const getIdeaByNftId = async ( ideaNft: number, decrypt: boolean = true, contract?: IContract, additionalFilters?: IIdeaByNftIdSearchParams, ): Promise<CONTRACT_INTERFACES.ITradeIdea | undefined> => { contract = ContractModule.thisOrGet(contract) let key = `getIdeaByNftId_${ideaNft}` if ( !isNullOrUndefined(additionalFilters?.strategyKey) && !isNullOrUndefined(additionalFilters?.ticker) && !isNullOrUndefined(additionalFilters?.type) ) { key = `getIdeaByNftId_${ideaNft}_${additionalFilters?.ticker}_${additionalFilters?.type}` } const { data } = await CacheService.getData({ contract: contract.gate!, key, init: async () => { let nftObject = await ProvableIdeasModule.general.getIdeaByNftId( ideaNft, decrypt, false, contract, ) const isPublic = getBoolean(nftObject.isPublic) let kind: CONTRACT_INTERFACES.ITradeIdeaIdeaKind | undefined = undefined let ticker: string | undefined = undefined if (isPublic) { kind = (nftObject.idea as CONTRACT_INTERFACES.ITradeIdeaIdea) .kind ticker = (nftObject.idea as CONTRACT_INTERFACES.ITradeIdeaIdea) .asset.ticker } if (nftObject) { nftObject.nftId = ideaNft ;(nftObject as any).unsave = true } if (isPublic || kind === 'close') { const type = isPublic ? 'public' : 'close' ;(nftObject as any).unsave = false await CacheService.updateData({ key: `getIdeaByNftId_${ideaNft}_${ticker}_${type}`, contract: contract?.gate!, data: nftObject, expirationInSeconds: skipExpirationInSeconds, forceToAdd: true, }) } if ( !isNullOrUndefined(additionalFilters?.strategyKey) && !isNullOrUndefined(additionalFilters?.ticker) && !isNullOrUndefined(additionalFilters?.type) ) { if ( additionalFilters?.strategyKey !== nftObject?.strategy?.reference || additionalFilters?.ticker !== (nftObject?.idea as any)?.asset?.ticker || additionalFilters?.type !== (isPublic ? 'public' : 'close') ) { return undefined } } return nftObject }, expirationInSeconds: skipExpirationInSeconds, }) return data } /* const filterToSkipLinkedIdeas = async ( result: IBasicNFT[], filterType: CONTRACT_INTERFACES.ITradeIdeaIdeaKind[] = [], ): Promise<IBasicNFT[]> => { const statusToFilter = kindTypeToIdeaStatus(filterType) if (filterType?.length <= 0) { filterType = ['open', 'adjust', 'close'] } const closedNfts = result.filter((each) => each.status === 3) const openOrAdjustIdeasCloseLinked: any[] = [] await Promise.all( closedNfts ?.map((res) => res.id) .map(async (nft) => { const restored = await getIdeaByNftId(nft) if (restored !== undefined) { for (const cci of ( restored!.idea as CONTRACT_INTERFACES.ITradeIdeaIdea ).chainIdeas!) { if (cci?.nftId) { openOrAdjustIdeasCloseLinked.push(cci?.nftId) } } } }), ) if (statusToFilter.length > 0) { result = result.filter((each) => statusToFilter.includes(each.status)) } const responseFiltered = [] for (const r of result) { if (r) { if (r?.status !== 3) { if (!openOrAdjustIdeasCloseLinked.includes(r?.id)) { responseFiltered.push(r) } } } } if (filterType.includes('close')) { responseFiltered.push(...closedNfts) } return responseFiltered } */ const ideaPublicVerificationHtml = async ( nftId: number, isPublic?: boolean, size?: 'small' | 'medium' | 'large', contract?: Contract, ): Promise<string> => { if (isNullOrUndefined(nftId) || nftId < 0) { throw new Error('nftId is required') } contract = ContractModule.thisOrGetGate(contract) let resposeMinified = null const minifyHtml = (html: string) => { return html .replace(/>\s+</g, '><') .replace(/\s{2,}/g, ' ') .replace(/>\s+/g, '>') .replace(/\s+</g, '<') } if (isNullOrUndefined(resposeMinified)) { let ideaDetails = { metadata: undefined as CONTRACT_INTERFACES.ITradeIdea | undefined, isPrivate: true, } try { ideaDetails.metadata = await getIdeaByNftId(nftId, true) } catch (err: any) { log.dev('ideaPublicVerificationHtml (ERROR)', err.message) } const metadata: CONTRACT_INTERFACES.ITradeIdea | undefined = ideaDetails.metadata const creator = metadata?.creator const strategy = metadata?.strategy const idea: CONTRACT_INTERFACES.ITradeIdeaIdea | undefined = metadata?.idea ? (metadata?.idea as CONTRACT_INTERFACES.ITradeIdeaIdea) : undefined const calcResultInfo = idea?.result const asset = idea?.asset const kind = idea?.kind const assetImage = idea?.asset.image log.dev('ideaPublicVerificationHtml (metadata)', metadata) const landingCheck = '' const mainArrow = '' const secondArrow = '' const stampBase64 = '' const styles = ` <style> @import url('https://fonts.googleapis.com/css2?family=Outfit:wght@200;400;600;800;900&display=swap'); /* LARGE */ .large-body { width: 850px !important; height: 271px !important; background: radial-gradient( 134.77% 134.77% at 47.29% 134.77%, rgba(202, 200, 233, 0.5) 0%, rgba(254, 254, 254, 0.5) 100% ) #ffffff !important; box-shadow: 0px -3px 57px rgba(0, 0, 0, 0.1) !important; border-radius: 20px !important; display: flex !important; align-items: center !important; padding: 30px !important; font-size: 20px !important; } .large-header { display: flex !important; justify-content: space-between !important; } .large-content { display: flex !important; align-items: flex-end !important; } .large-type { display: flex !important; align-items: center !important; } .large-text { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 0.7em !important; line-height: 1.4em !important; color: rgba(0, 0, 0, 0.5) !important; margin: 0px !important; max-height: 52.88px !important; text-overflow: ellipsis !important; overflow: hidden !important; } .large-name-container { display: flex !important; align-items: center !important; } .large-name { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.4em !important; line-height: 1.75em !important; margin: 0 15px !important; } .large-abrev { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 1.4em !important; line-height: 1.75em !important; } .by-label { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 0.7em !important; line-height: 0.9em !important; } .by { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 400 !important; font-size: 0.7em !important; line-height: 0.9em !important; } .large-stamp { margin-right: 1.75em !important; } .large-position { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 300 !important; font-size: 0.8em !important; line-height: 1em !important; color: rgba(0, 0, 0, 0.5) !important; } .large-button { background: radial-gradient( 110.14% 356.77% at -8.3% -103.85%, #8d61eb 0%, #24a1d4 100% ) !important; border-radius: 26.822px !important; color: white !important; font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1em !important; line-height: 1.2em !important; padding: 6.25px 21px !important; border: none !important; display: flex !important; align-items: center !important; justify-content: center !important; margin-left: 1.75em !important; } .large-button:hover { cursor: pointer !important; } .large-button span { margin-right: 5px !important; } .large-duration { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 500 !important; font-size: 0.8em !important; line-height: 1em !important; color: #6d71e7 !important; padding: 5.32px 17px !important; background: rgba(109, 113, 231, 0.1) !important; border-radius: 8.66899px !important; } .large-percent { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.35em !important; line-height: 1.155em !important; } .large-profit { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 400 !important; font-size: 1em !important; line-height: 1.25em !important; } .large-access { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.4em !important; line-height: 1.65em !important; } .large-access-text { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 300 !important; font-size: 0.8em !important; line-height: 1em !important; color: rgba(0, 0, 0, 0.5) !important; } .large-logo { width: 36px !important; height: 36px !important; border-radius: 50% !important; } .large-arrows { margin: 0 2.5em !important; } .large-right { display: flex !important; flex-direction: column !important; justify-content: space-between !important; height: 211px !important; text-align: initial !important; } .noaccess { justify-content: space-around !important; padding: 35px 0 !important; text-align: initial !important; } .text-noaccess { font-size: 110% !important; } /* LARGE END */ /* MEDIUM */ .medium-body { width: 525px !important; height: 167px !important; background: radial-gradient( 134.77% 134.77% at 47.29% 134.77%, rgba(202, 200, 233, 0.5) 0%, rgba(254, 254, 254, 0.5) 100% ) #ffffff !important; box-shadow: 0px -3px 57px rgba(0, 0, 0, 0.1) !important; border-radius: 20px !important; display: flex !important; align-items: center !important; padding: 18.5px !important; font-size: 12.35px !important; } .medium-header { display: flex !important; justify-content: space-between !important; } .medium-content { display: flex !important; align-items: flex-end !important; } .medium-type { display: flex !important; align-items: center !important; } .medium-text { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 0.7em !important; line-height: 1.4em !important; color: rgba(0, 0, 0, 0.5) !important; margin: 0px !important; max-height: 52.88px !important; text-overflow: ellipsis !important; overflow: hidden !important; } .medium-name-container { display: flex !important; align-items: center !important; } .medium-name { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.4em !important; line-height: 1.75em !important; margin: 0 10px !important; } .medium-abrev { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 1.4em !important; line-height: 1.75em !important; } .by-label { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 0.7em !important; line-height: 0.9em !important; } .by { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 400 !important; font-size: 0.7em !important; line-height: 0.9em !important; } .medium-stamp { width: 110px !important; margin-right: 10px !important; } .medium-position { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 300 !important; font-size: 0.8em !important; line-height: 1em !important; color: rgba(0, 0, 0, 0.5) !important; } .medium-button { background: radial-gradient( 110.14% 356.77% at -8.3% -103.85%, #8d61eb 0%, #24a1d4 100% ) !important; border-radius: 26.822px !important; color: white !important; font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1em !important; line-height: 1.2em !important; padding: 6.25px 21px !important; border: none !important; display: flex !important; align-items: center !important; justify-content: center !important; margin-left: 1.75em !important; } .medium-button:hover { cursor: pointer !important; } .medium-button span { margin-right: 5px !important; } .medium-duration { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 500 !important; font-size: 0.8em !important; line-height: 1em !important; color: #6d71e7 !important; padding: 5.32px 17px !important; background: rgba(109, 113, 231, 0.1) !important; border-radius: 8.66899px !important; } .medium-percent { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.35em !important; line-height: 1.155em !important; } .medium-profit { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 400 !important; font-size: 1em !important; line-height: 1.25em !important; } .medium-access { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.4em !important; line-height: 1.65em !important; } .medium-access-text { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 300 !important; font-size: 0.8em !important; line-height: 1em !important; color: rgba(0, 0, 0, 0.5) !important; } .medium-logo { width: 26px !important; height: 26px !important; border-radius: 50% !important; } .medium-arrows { width: 13px !important; margin: 0 2em !important; } .medium-right { display: flex !important; flex-direction: column !important; justify-content: space-between !important; height: 130px !important; text-align: initial !important; } .noaccess { justify-content: space-around !important; padding: 35px 0 !important; text-align: initial !important; } .text-noaccess { font-size: 110% !important; } /* MEDIUM END */ /* SMALL */ .small-body { width: 300px !important; height: 120px !important; background: radial-gradient( 134.77% 134.77% at 47.29% 134.77%, rgba(202, 200, 233, 0.5) 0%, rgba(254, 254, 254, 0.5) 100% ) #ffffff !important; box-shadow: 0px -3px 57px rgba(0, 0, 0, 0.1) !important; border-radius: 20px !important; display: flex !important; align-items: center !important; padding: 10px !important; font-size: 12.35px !important; } .small-header { display: flex !important; flex-direction: column !important; } .small-content { display: flex !important; align-items: flex-end !important; } .small-type { display: flex !important; align-items: center !important; } .small-text { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 0.7em !important; line-height: 1.4em !important; color: rgba(0, 0, 0, 0.5) !important; margin: 0px !important; max-height: 52.88px !important; text-overflow: ellipsis !important; overflow: hidden !important; } .small-name-container { display: flex !important; align-items: center !important; } .small-name { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.4em !important; line-height: 1.75em !important; margin: 0 5px !important; } .small-abrev { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 1.4em !important; line-height: 1.75em !important; } .by-label { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 200 !important; font-size: 0.7em !important; line-height: 0.9em !important; } .by { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 400 !important; font-size: 0.7em !important; line-height: 0.9em !important; } .small-stamp { width: 80px !important; margin-right: 10px !important; } .small-position { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 300 !important; font-size: 0.8em !important; line-height: 1em !important; color: rgba(0, 0, 0, 0.5) !important; } .small-button { background: radial-gradient( 110.14% 356.77% at -8.3% -103.85%, #8d61eb 0%, #24a1d4 100% ) !important; border-radius: 26.822px !important; color: white !important; font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1em !important; line-height: 1.2em !important; padding: 6.25px 21px !important; border: none !important; display: flex !important; align-items: center !important; justify-content: center !important; margin-left: 1.75em !important; } .small-button:hover { cursor: pointer !important; } .small-button span { margin-right: 5px !important; } .small-duration { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 500 !important; font-size: 0.8em !important; line-height: 1em !important; color: #6d71e7 !important; padding: 5.32px 17px !important; background: rgba(109, 113, 231, 0.1) !important; border-radius: 8.66899px !important; } .small-percent { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.35em !important; line-height: 1.155em !important; } .small-profit { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 400 !important; font-size: 1em !important; line-height: 1.25em !important; } .small-access { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 700 !important; font-size: 1.4em !important; line-height: 1.65em !important; } .small-access-text { font-family: "Outfit", sans-serif !important; font-style: normal !important; font-weight: 300 !important; font-size: 0.8em !important; line-height: 1em !important; color: rgba(0, 0, 0, 0.5) !important; } .small-logo { width: 20px !important; height: 20px !important; border-radius: 50% !important; } .small-arrows { width: 13px !important; margin: 0 1em !important; } .small-right { display: flex !important; flex-direction: column !important; justify-content: space-between !important; height: 100px !important; text-align: initial !important; } .noaccess { justify-content: space-around !important; padding: 35px 0 !important; text-align: initial !important; } .text-noaccess { font-size: 110% !important; } .hidden { display: none !important; } /* SMALL END */ </style> ` const sizePrefix = size || 'small' let htmlContent = '' if (kind !== 'close') { htmlContent = ` ${styles} <div class="${sizePrefix}-body"> <div class="${sizePrefix}-left"> <img src="${stampBase64}" alt="" class="${sizePrefix}-stamp" /> </div> <div class="${sizePrefix}-right noaccess"> <span class="${sizePrefix}-access">INSUFFICIENT ACCESS</span> <p class="${sizePrefix}-access-text text-noaccess"> Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Et malesuada fames ac turpis egestas maecenas. Felis donec et odio pellentesque diam volutpat commodo sed. </p> </div> </div>` } if (kind === 'close') { htmlContent = ` ${styles} <div class="${sizePrefix}-body"> <div class="${sizePrefix}-left"> <img src="${stampBase64}" alt="" class="${sizePrefix}-stamp" /> </div> <div class="${sizePrefix}-right"> <div class="${sizePrefix}-header"> <div class="${sizePrefix}-name-container"> <img src="${assetImage}" alt="" class="${sizePrefix}-logo" /> <span class="${sizePrefix}-name">${asset?.ticker}</span> <span class="${sizePrefix}-abrev">| ${asset?.description}</span> </div> <div class="by-container"> <span class="by-label">BY&nbsp;</span> <span class="by">${ strategy?.name?.toUpperCase() || creator?.name?.toUpperCase() }&nbsp;</span> <img src="${landingCheck}" alt="" class="by-img" /> </div> </div> <div class="${sizePrefix}-position"> Position Opened at $${ calcResultInfo?.calculation[0].price } and closed at $${ calcResultInfo?.calculation[ calcResultInfo?.calculation.length - 1 ].price } </div> <div class="${sizePrefix}-type"> <span class="${sizePrefix}-duration">${ calcResultInfo?.summary.profit || calcResultInfo?.summary.percentage === 0 ? 'LONG' : 'SHORT' }</span> <img src="${mainArrow}" alt="" class="${sizePrefix}-arrows" /> <div> <span class="${sizePrefix}-percent">${calcResultInfo?.summary.percentage.toFixed( 6, )}%</span> <span class="${sizePrefix}-profit">&nbsp;PROFIT</span> </div> </div> <div class="${sizePrefix}-content"> <div class="${sizePrefix}-content-left"> <p class="${sizePrefix}-text"> This trade idea was created as an NFT and saved on a blockchain so that the price at which it was opened and closed can be independently verified after the fact </p> </div> <div class="${sizePrefix}-content-right"> <button class="${sizePrefix}-button"> <span>VIEW</span> <img src="${secondArrow}" alt="" /> </button> </div> </div> </div> </div> ` } resposeMinified = minifyHtml(htmlContent) } return resposeMinified as string } const getProviderList = async (): Promise<string[]> => { return ['0x2767441E044aCd9bbC21a759fB0517494875092d'] } const getPricingProviderList = async () => { const providers = [...CONTRACT_CONSTANTS.PRICING_PROVIDER] return providers.map((provider) => { return { name: provider, // fullName: provider, enabled: true, } }) } const resetCache = async (): Promise<void> => { await CacheService.resetData() } export const MigrationV3TemporaryModule = { resetCache, getETHBalance, ideaPublicVerificationHtml, getProviderList, getPricingProviderList, isCloseIdea, }