Testnet and mainnet entries both dynamic

This commit is contained in:
Mentor Palokaj 2021-10-19 12:23:18 +02:00
parent 42aa39c343
commit 63d72cbfcb
5 changed files with 2784 additions and 2570 deletions

View File

@ -1,60 +1,7 @@
const app = require( './express' )()
const name = require( 'random-name' )
const { db } = require( './firebase' )
const { getTotalSupply } = require( './contract' )
const { safelyReturnRocketeer, web2domain } = require( './rocketeer' )
// ///////////////////////////////
// Data sources
// ///////////////////////////////
const globalAttributes = [
{ trait_type: "Age", display_type: "number", values: [
{ value: 35, probability: .5 },
{ value: 45, probability: .25 },
{ value: 25, probability: .25 }
] }
]
const heavenlyBodies = [ "Mercury", "Venus", "Earth", "Mars", "Jupiter", "Saturn", "Uranus", "Neptune", "Pluto", "the Moon", "the Sun" ]
const web2domain = 'https://rocketeer.fans'
// ///////////////////////////////
// Rocketeer helpers
// ///////////////////////////////
// Pick random item from array with equal probability
const pickRandomArrayEntry = array => array[ Math.floor( Math.random() * array.length ) ]
// Pick random attributes based on global attribute array
function pickRandomAttributes( attributes ) {
// Decimal accuracy, if probabilities have the lowest 0.01 then 100 is enough, for 0.001 1000 is needed
const probabilityDecimals = 3
// Remap the trait so it has a 'lottery ticket box' based on probs
const attributeLottery = attributes.map( ( { values, ...attribute } ) => ( {
// Attribute meta stays the same
...attribute,
// Values are reduced from objects with probabilities to an array with elements
values: values.reduce( ( acc, val ) => {
const { probability, value } = val
// Map probabilities to a flat array of items
const amountToAdd = 10 * probabilityDecimals * probability
for ( let i = 0; i < amountToAdd; i++ ) acc.push( value )
return acc
}, [] )
} ) )
// Pick a random element from the lottery box array items
return attributeLottery.map( ( { values, ...attribute } ) => ( {
// Attribute meta stays the same
...attribute,
// Select random entry from array
value: pickRandomArrayEntry( values )
} ) )
}
// ///////////////////////////////
// Specific Rocketeer instances
@ -65,74 +12,32 @@ app.get( '/api/rocketeer/:id', async ( req, res ) => {
const { id } = req.params
if( !id ) return res.json( { error: `No ID specified in URL` } )
// Chech if this is an illegal ID
try {
// Get the last know total supply
const { cachedTotalSupply } = await db.collection( 'meta' ).doc( 'contract' ).get().then( doc => doc.data() )
// Get old rocketeer if it exists
const rocketeer = await safelyReturnRocketeer( id, 'mainnet' )
// If the requested ID is larger than that, check if the new total supply is more
if( cachedTotalSupply < id ) {
// Get net total supply through infura, if infura fails, return the cached value just in case
const totalSupply = await getTotalSupply().catch( f => cachedTotalSupply )
// Write new value to cache
await db.collection( 'meta' ).doc( 'contract' ).set( { cachedTotalSupply: totalSupply }, { merge: true } )
// If the requested ID is larger than total supply, exit
if( totalSupply < id ) return res.json( {
trace: 'total supply getter',
error: 'This Rocketeer does not yet exist.'
} )
}
} catch( e ) {
return res.json( { trace: 'total supply getter', error: e.message || JSON.stringify( e ) } )
}
// Get existing rocketeer if it exists
try {
const oldRocketeer = await db.collection( 'rocketeers' ).doc( id ).get().then( doc => doc.data() )
if( oldRocketeer ) return res.json( oldRocketeer )
// Return the new rocketeer
return res.json( rocketeer )
} catch( e ) {
return res.json( { trace: 'firestore rocketeer read',error: e.message || JSON.stringify( e ) } )
// Log error for debugging
console.error( `Mainnet api error for ${ id }: `, e )
// Return error to frontend
return res.json( { error: e.mesage || e.toString() } )
}
// The base object of a new Rocketeer
const rocketeer = {
name: `${ name.first() } ${ name.middle() } ${ name.last() } of ${ pickRandomArrayEntry( heavenlyBodies ) }`,
description: ``,
image: ``,
external_url: `${ web2domain }/api/rocketeer/${ id }`,
attributes: []
}
// Generate randomized attributes
rocketeer.attributes = pickRandomAttributes( globalAttributes )
// TODO: Generate, compile and upload image
rocketeer.image = web2domain
// Save new Rocketeer
try {
await db.collection( 'rocketeers' ).doc( id ).set( rocketeer )
} catch( e ) {
return res.json( { trace: 'firestore rocketeer save', error: e.message || JSON.stringify( e ) } )
}
// Return the new rocketeer
return res.json( rocketeer )
} )
// ///////////////////////////////
// Static collection data
// ///////////////////////////////
app.get( '/api/collection', ( req, res ) => res.json( {
app.get( '/api/collection', async ( req, res ) => res.json( {
totalSupply: await getTotalSupply( 'mainnet' ).catch( f => 'error' ),
description: "A testnet collection",
external_url: web2domain,
image: "https://rocketpool.net/images/rocket.png",

View File

@ -0,0 +1,143 @@
const name = require( 'random-name' )
const { db } = require( './firebase' )
const { getTotalSupply } = require( './contract' )
// ///////////////////////////////
// Data sources
// ///////////////////////////////
const globalAttributes = [
{ trait_type: "Age", display_type: "number", values: [
{ value: 35, probability: .5 },
{ value: 45, probability: .25 },
{ value: 25, probability: .25 }
] }
]
const heavenlyBodies = [ "Mercury", "Venus", "Earth", "Mars", "Jupiter", "Saturn", "Uranus", "Neptune", "Pluto", "the Moon", "the Sun" ]
const web2domain = 'https://rocketeer.fans'
// ///////////////////////////////
// Rocketeer helpers
// ///////////////////////////////
// Pick random item from array with equal probability
const pickRandomArrayEntry = array => array[ Math.floor( Math.random() * array.length ) ]
// Pick random attributes based on global attribute array
function pickRandomAttributes( attributes ) {
// Decimal accuracy, if probabilities have the lowest 0.01 then 100 is enough, for 0.001 1000 is needed
const probabilityDecimals = 3
// Remap the trait so it has a 'lottery ticket box' based on probs
const attributeLottery = attributes.map( ( { values, ...attribute } ) => ( {
// Attribute meta stays the same
...attribute,
// Values are reduced from objects with probabilities to an array with elements
values: values.reduce( ( acc, val ) => {
const { probability, value } = val
// Map probabilities to a flat array of items
const amountToAdd = 10 * probabilityDecimals * probability
for ( let i = 0; i < amountToAdd; i++ ) acc.push( value )
return acc
}, [] )
} ) )
// Pick a random element from the lottery box array items
return attributeLottery.map( ( { values, ...attribute } ) => ( {
// Attribute meta stays the same
...attribute,
// Select random entry from array
value: pickRandomArrayEntry( values )
} ) )
}
// ///////////////////////////////
// Caching
// ///////////////////////////////
async function isInvalidRocketeerId( id, network='mainnet' ) {
// Chech if this is an illegal ID
try {
// Get the last know total supply
const { cachedTotalSupply } = await db.collection( 'meta' ).doc( network ).get().then( doc => doc.data() ) || {}
// If the requested ID is larger than that, check if the new total supply is more
if( !cachedTotalSupply || cachedTotalSupply < id ) {
// Get net total supply through infura, if infura fails, return the cached value just in case
const totalSupply = await getTotalSupply( network )
// Write new value to cache
await db.collection( 'meta' ).doc( network ).set( { cachedTotalSupply: totalSupply }, { merge: true } )
// If the requested ID is larger than total supply, exit
if( totalSupply < id ) throw new Error( `Invalid ID ${ id }, total supply is ${ totalSupply }` )
// If all good, return true
return false
}
} catch( e ) {
return e
}
}
async function getExistingRocketeer( id, network='mainnet' ) {
return db.collection( `${ network }Rocketeers` ).doc( id ).get().then( doc => doc.data() ).catch( f => false )
}
// ///////////////////////////////
// Rocketeer generator
// ///////////////////////////////
async function generateRocketeer( id, network='mainnet' ) {
// The base object of a new Rocketeer
const rocketeer = {
name: `${ name.first() } ${ name.middle() } ${ name.last() } of ${ pickRandomArrayEntry( heavenlyBodies ) }`,
description: ``,
image: ``,
external_url: `${ web2domain }/${ network == 'mainnet' ? 'api' : 'testnetapi' }/rocketeer/${ id }`,
attributes: []
}
// Generate randomized attributes
rocketeer.attributes = pickRandomAttributes( globalAttributes )
// TODO: Generate, compile and upload image
rocketeer.image = web2domain
// Save new Rocketeer
await db.collection( `${ network }Rocketeers` ).doc( id ).set( rocketeer )
return rocketeer
}
async function safelyReturnRocketeer( id, network ) {
// Chech if this is an illegal ID
const invalidId = await isInvalidRocketeerId( id, network )
if( invalidId ) throw invalidId
// Get old rocketeer if it exists
const oldRocketeer = await getExistingRocketeer( id, network )
if( oldRocketeer ) return oldRocketeer
// If no old rocketeer exists, make a new one and save it
return generateRocketeer( id, network )
}
module.exports = {
web2domain: web2domain,
safelyReturnRocketeer: safelyReturnRocketeer
}

View File

@ -1,24 +1,40 @@
const app = require( './express' )()
const { getTotalSupply } = require( './contract' )
const { safelyReturnRocketeer, web2domain } = require( './rocketeer' )
// Specific Rocketeer instances
app.get( '/testnetapi/rocketeer/:id', async ( req, res ) => res.json( {
description: "A testnet Rocketeer",
external_url: `https://openseacreatures.io/${ req.params.id }`,
image: "https://rocketpool.net/images/rocket.png",
name: `Rocketeer number ${ req.params.id }`,
attributes: [
{ trait_type: "Occupation", value: "Rocketeer" },
{ trait_type: "Age", display_type: "number", value: req.params.id + 42 }
]
} ) )
app.get( '/testnetapi/rocketeer/:id', async ( req, res ) => {
// Parse the request
const { id } = req.params
if( !id ) return res.json( { error: `No ID specified in URL` } )
try {
// Get old rocketeer if it exists
const rocketeer = await safelyReturnRocketeer( id, 'rinkeby' )
// Return the new rocketeer
return res.json( rocketeer )
} catch( e ) {
// Log error for debugging
console.error( `Testnet api error for ${ id }: `, Object.keys( e ) )
// Return error to frontend
return res.json( { error: e.mesage || e.toString() } )
}
} )
// Collection data
app.get( '/testnetapi/collection', async ( req, res ) => res.json( {
totalSupply: await getTotalSupply( 'rinkeby' ),
totalSupply: await getTotalSupply( 'rinkeby' ).catch( f => 'error' ),
description: "A testnet collection",
external_url: `https://openseacreatures.io/`,
external_url: web2domain,
image: "https://rocketpool.net/images/rocket.png",
name: `Rocketeer collection`,
seller_fee_basis_points: 0,

5018
website/package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -20,41 +20,41 @@
},
"homepage": "https://github.com/actuallymentor/static-webpage-boilerplate-webpack-browsersync#readme",
"dependencies": {
"@babel/core": "^7.13.15",
"@babel/core": "^7.15.8",
"@babel/polyfill": "^7.12.1",
"@babel/preset-env": "^7.13.15",
"autoprefixer": "^10.2.5",
"@babel/preset-env": "^7.15.8",
"autoprefixer": "^10.3.7",
"babel-loader": "^8.2.2",
"browser-sync": "^2.26.14",
"browser-sync": "^2.27.5",
"browser-sync-webpack-plugin": "^2.3.0",
"css-loader": "^5.2.1",
"cssnano": "^5.0.0",
"css-loader": "^5.2.7",
"cssnano": "^5.0.8",
"del": "^6.0.0",
"doiuse": "^4.4.1",
"dotenv": "^8.2.0",
"dotenv": "^8.6.0",
"html-minifier": "^4.0.0",
"ip": "^1.1.5",
"mkdirp": "^1.0.4",
"ncp": "^2.0.0",
"node-sass": "^5.0.0",
"postcss": "^8.2.10",
"postcss": "^8.3.9",
"pug": "^3.0.2",
"sharp": "^0.28.1",
"sharp": "^0.29.1",
"sitemap": "^6.4.0",
"webpack": "^5.31.2"
"webpack": "^5.58.2"
},
"devDependencies": {
"@babel/eslint-parser": "^7.13.14",
"@babel/plugin-transform-runtime": "^7.13.15",
"@babel/eslint-parser": "^7.15.8",
"@babel/plugin-transform-runtime": "^7.15.8",
"chai": "^4.3.4",
"chai-as-promised": "^7.1.1",
"eslint": "^7.24.0",
"eslint": "^7.32.0",
"esm": "^3.2.25",
"mocha": "^8.3.2",
"mocha": "^8.4.0",
"recursive-readdir": "^2.2.2",
"request": "^2.88.2",
"request-promise-native": "^1.0.9",
"webpack-cli": "^4.6.0"
"webpack-cli": "^4.9.1"
},
"optionalDependencies": {
"fsevents": "^2.3.2"