@ixily/activ
Version:
Alpha Capture Trade Idea Verification. Blockchain ownership proven trade ideas and strategies.
904 lines (848 loc) • 39.7 kB
text/typescript
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 ;
height: 271px ;
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 ;
box-shadow: 0px -3px 57px rgba(0, 0, 0, 0.1) ;
border-radius: 20px ;
display: flex ;
align-items: center ;
padding: 30px ;
font-size: 20px ;
}
.large-header {
display: flex ;
justify-content: space-between ;
}
.large-content {
display: flex ;
align-items: flex-end ;
}
.large-type {
display: flex ;
align-items: center ;
}
.large-text {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 0.7em ;
line-height: 1.4em ;
color: rgba(0, 0, 0, 0.5) ;
margin: 0px ;
max-height: 52.88px ;
text-overflow: ellipsis ;
overflow: hidden ;
}
.large-name-container {
display: flex ;
align-items: center ;
}
.large-name {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.4em ;
line-height: 1.75em ;
margin: 0 15px ;
}
.large-abrev {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 1.4em ;
line-height: 1.75em ;
}
.by-label {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 0.7em ;
line-height: 0.9em ;
}
.by {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 400 ;
font-size: 0.7em ;
line-height: 0.9em ;
}
.large-stamp {
margin-right: 1.75em ;
}
.large-position {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 300 ;
font-size: 0.8em ;
line-height: 1em ;
color: rgba(0, 0, 0, 0.5) ;
}
.large-button {
background: radial-gradient(
110.14% 356.77% at -8.3% -103.85%,
#8d61eb 0%,
#24a1d4 100%
) ;
border-radius: 26.822px ;
color: white ;
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1em ;
line-height: 1.2em ;
padding: 6.25px 21px ;
border: none ;
display: flex ;
align-items: center ;
justify-content: center ;
margin-left: 1.75em ;
}
.large-button:hover {
cursor: pointer ;
}
.large-button span {
margin-right: 5px ;
}
.large-duration {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 500 ;
font-size: 0.8em ;
line-height: 1em ;
color: #6d71e7 ;
padding: 5.32px 17px ;
background: rgba(109, 113, 231, 0.1) ;
border-radius: 8.66899px ;
}
.large-percent {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.35em ;
line-height: 1.155em ;
}
.large-profit {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 400 ;
font-size: 1em ;
line-height: 1.25em ;
}
.large-access {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.4em ;
line-height: 1.65em ;
}
.large-access-text {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 300 ;
font-size: 0.8em ;
line-height: 1em ;
color: rgba(0, 0, 0, 0.5) ;
}
.large-logo {
width: 36px ;
height: 36px ;
border-radius: 50% ;
}
.large-arrows {
margin: 0 2.5em ;
}
.large-right {
display: flex ;
flex-direction: column ;
justify-content: space-between ;
height: 211px ;
text-align: initial ;
}
.noaccess {
justify-content: space-around ;
padding: 35px 0 ;
text-align: initial ;
}
.text-noaccess {
font-size: 110% ;
}
/* LARGE END */
/* MEDIUM */
.medium-body {
width: 525px ;
height: 167px ;
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 ;
box-shadow: 0px -3px 57px rgba(0, 0, 0, 0.1) ;
border-radius: 20px ;
display: flex ;
align-items: center ;
padding: 18.5px ;
font-size: 12.35px ;
}
.medium-header {
display: flex ;
justify-content: space-between ;
}
.medium-content {
display: flex ;
align-items: flex-end ;
}
.medium-type {
display: flex ;
align-items: center ;
}
.medium-text {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 0.7em ;
line-height: 1.4em ;
color: rgba(0, 0, 0, 0.5) ;
margin: 0px ;
max-height: 52.88px ;
text-overflow: ellipsis ;
overflow: hidden ;
}
.medium-name-container {
display: flex ;
align-items: center ;
}
.medium-name {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.4em ;
line-height: 1.75em ;
margin: 0 10px ;
}
.medium-abrev {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 1.4em ;
line-height: 1.75em ;
}
.by-label {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 0.7em ;
line-height: 0.9em ;
}
.by {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 400 ;
font-size: 0.7em ;
line-height: 0.9em ;
}
.medium-stamp {
width: 110px ;
margin-right: 10px ;
}
.medium-position {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 300 ;
font-size: 0.8em ;
line-height: 1em ;
color: rgba(0, 0, 0, 0.5) ;
}
.medium-button {
background: radial-gradient(
110.14% 356.77% at -8.3% -103.85%,
#8d61eb 0%,
#24a1d4 100%
) ;
border-radius: 26.822px ;
color: white ;
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1em ;
line-height: 1.2em ;
padding: 6.25px 21px ;
border: none ;
display: flex ;
align-items: center ;
justify-content: center ;
margin-left: 1.75em ;
}
.medium-button:hover {
cursor: pointer ;
}
.medium-button span {
margin-right: 5px ;
}
.medium-duration {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 500 ;
font-size: 0.8em ;
line-height: 1em ;
color: #6d71e7 ;
padding: 5.32px 17px ;
background: rgba(109, 113, 231, 0.1) ;
border-radius: 8.66899px ;
}
.medium-percent {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.35em ;
line-height: 1.155em ;
}
.medium-profit {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 400 ;
font-size: 1em ;
line-height: 1.25em ;
}
.medium-access {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.4em ;
line-height: 1.65em ;
}
.medium-access-text {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 300 ;
font-size: 0.8em ;
line-height: 1em ;
color: rgba(0, 0, 0, 0.5) ;
}
.medium-logo {
width: 26px ;
height: 26px ;
border-radius: 50% ;
}
.medium-arrows {
width: 13px ;
margin: 0 2em ;
}
.medium-right {
display: flex ;
flex-direction: column ;
justify-content: space-between ;
height: 130px ;
text-align: initial ;
}
.noaccess {
justify-content: space-around ;
padding: 35px 0 ;
text-align: initial ;
}
.text-noaccess {
font-size: 110% ;
}
/* MEDIUM END */
/* SMALL */
.small-body {
width: 300px ;
height: 120px ;
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 ;
box-shadow: 0px -3px 57px rgba(0, 0, 0, 0.1) ;
border-radius: 20px ;
display: flex ;
align-items: center ;
padding: 10px ;
font-size: 12.35px ;
}
.small-header {
display: flex ;
flex-direction: column ;
}
.small-content {
display: flex ;
align-items: flex-end ;
}
.small-type {
display: flex ;
align-items: center ;
}
.small-text {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 0.7em ;
line-height: 1.4em ;
color: rgba(0, 0, 0, 0.5) ;
margin: 0px ;
max-height: 52.88px ;
text-overflow: ellipsis ;
overflow: hidden ;
}
.small-name-container {
display: flex ;
align-items: center ;
}
.small-name {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.4em ;
line-height: 1.75em ;
margin: 0 5px ;
}
.small-abrev {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 1.4em ;
line-height: 1.75em ;
}
.by-label {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 200 ;
font-size: 0.7em ;
line-height: 0.9em ;
}
.by {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 400 ;
font-size: 0.7em ;
line-height: 0.9em ;
}
.small-stamp {
width: 80px ;
margin-right: 10px ;
}
.small-position {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 300 ;
font-size: 0.8em ;
line-height: 1em ;
color: rgba(0, 0, 0, 0.5) ;
}
.small-button {
background: radial-gradient(
110.14% 356.77% at -8.3% -103.85%,
#8d61eb 0%,
#24a1d4 100%
) ;
border-radius: 26.822px ;
color: white ;
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1em ;
line-height: 1.2em ;
padding: 6.25px 21px ;
border: none ;
display: flex ;
align-items: center ;
justify-content: center ;
margin-left: 1.75em ;
}
.small-button:hover {
cursor: pointer ;
}
.small-button span {
margin-right: 5px ;
}
.small-duration {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 500 ;
font-size: 0.8em ;
line-height: 1em ;
color: #6d71e7 ;
padding: 5.32px 17px ;
background: rgba(109, 113, 231, 0.1) ;
border-radius: 8.66899px ;
}
.small-percent {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.35em ;
line-height: 1.155em ;
}
.small-profit {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 400 ;
font-size: 1em ;
line-height: 1.25em ;
}
.small-access {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 700 ;
font-size: 1.4em ;
line-height: 1.65em ;
}
.small-access-text {
font-family: "Outfit", sans-serif ;
font-style: normal ;
font-weight: 300 ;
font-size: 0.8em ;
line-height: 1em ;
color: rgba(0, 0, 0, 0.5) ;
}
.small-logo {
width: 20px ;
height: 20px ;
border-radius: 50% ;
}
.small-arrows {
width: 13px ;
margin: 0 1em ;
}
.small-right {
display: flex ;
flex-direction: column ;
justify-content: space-between ;
height: 100px ;
text-align: initial ;
}
.noaccess {
justify-content: space-around ;
padding: 35px 0 ;
text-align: initial ;
}
.text-noaccess {
font-size: 110% ;
}
.hidden {
display: none ;
}
/* 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 </span>
<span class="by">${
strategy?.name?.toUpperCase() || creator?.name?.toUpperCase()
} </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"> 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,
}