mirror of
				https://github.com/Instadapp/trustwallet-assets.git
				synced 2024-07-29 22:37:31 +00:00 
			
		
		
		
	 5efd533fdc
			
		
	
	
		5efd533fdc
		
			
		
	
	
	
	
		
			
			* Extra logs * External Updates * Move token validity check, cheksumFormat ID is available only later. * External Updates Co-authored-by: Catenocrypt <catenocrypt@users.noreply.github.com>
		
			
				
	
	
		
			265 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			265 lines
		
	
	
		
			8.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| // Tokenlist.json handling
 | |
| 
 | |
| import { readJsonFile, writeJsonFile } from "../generic/json";
 | |
| import { diff } from "jsondiffpatch";
 | |
| import { tokenInfoFromTwApi, TokenTwInfo } from "../generic/asset";
 | |
| import {
 | |
|     getChainAssetLogoPath,
 | |
|     getChainAllowlistPath,
 | |
|     getChainTokenlistBasePath,
 | |
|     getChainTokenlistPath,
 | |
| } from "../generic/repo-structure";
 | |
| import * as bluebird from "bluebird";
 | |
| import { isPathExistsSync } from "../generic/filesystem";
 | |
| 
 | |
| class Version {
 | |
|     major: number
 | |
|     minor: number
 | |
|     patch: number
 | |
| 
 | |
|     constructor(major: number, minor: number, patch: number) {
 | |
|         this.major = major
 | |
|         this.minor = minor
 | |
|         this.patch = patch
 | |
|     }
 | |
| }
 | |
| 
 | |
| export class List {
 | |
|     name: string
 | |
|     logoURI: string
 | |
|     timestamp: string
 | |
|     tokens: TokenItem[]
 | |
|     pairs: Pair[]
 | |
|     version: Version
 | |
| 
 | |
|     constructor(name: string, logoURI: string, timestamp: string, tokens: TokenItem[], version: Version) {
 | |
|         this.name = name
 | |
|         this.logoURI = logoURI
 | |
|         this.timestamp = timestamp;
 | |
|         this.tokens = tokens
 | |
|         this.version = version
 | |
|     }
 | |
| }
 | |
| 
 | |
| export class TokenItem {
 | |
|     asset: string;
 | |
|     type: string;
 | |
|     address: string;
 | |
|     name: string;
 | |
|     symbol: string;
 | |
|     decimals: number;
 | |
|     logoURI: string;
 | |
|     pairs: Pair[];
 | |
| 
 | |
|     constructor(asset: string, type: string, address: string, name: string, symbol: string, decimals: number, logoURI: string, pairs: Pair[]) {
 | |
|         this.asset = asset
 | |
|         this.type = type
 | |
|         this.address = address
 | |
|         this.name = name;
 | |
|         this.symbol = symbol
 | |
|         this.decimals = decimals
 | |
|         this.logoURI = logoURI
 | |
|         this.pairs = pairs
 | |
|     }
 | |
| }
 | |
| 
 | |
| export class Pair {
 | |
|     base: string;
 | |
|     lotSize?: string;
 | |
|     tickSize?: string;
 | |
| 
 | |
|     constructor(base: string, lotSize?: string, tickSize?: string) {
 | |
|         this.base = base
 | |
|         this.lotSize = lotSize
 | |
|         this.tickSize = tickSize
 | |
|     }
 | |
| }
 | |
| 
 | |
| export function createTokensList(titleCoin: string, tokens: TokenItem[], time: string, versionMajor: number, versionMinor = 1, versionPatch = 0): List {
 | |
|     if (!time) {
 | |
|         time = (new Date()).toISOString();
 | |
|     }
 | |
|     const list = new List(
 | |
|         `Trust Wallet: ${titleCoin}`,
 | |
|         "https://trustwallet.com/assets/images/favicon.png",
 | |
|         time,
 | |
|         tokens,
 | |
|         new Version(versionMajor, versionMinor, versionPatch)
 | |
|     );
 | |
|     sort(list);
 | |
|     return list;
 | |
| }
 | |
| 
 | |
| function totalPairs(list: List): number {
 | |
|     let c = 0;
 | |
|     list.tokens.forEach(t => c += (t.pairs || []).length);
 | |
|     return c;
 | |
| }
 | |
| 
 | |
| export function writeToFile(filename: string, list: List): void {
 | |
|     writeJsonFile(filename, list);
 | |
|     console.log(`Tokenlist: list with ${list.tokens.length} tokens and ${totalPairs(list)} pairs written to ${filename}.`);
 | |
| }
 | |
| 
 | |
| // Write out to file, updating version+timestamp if there was change
 | |
| export function writeToFileWithUpdate(filename: string, list: List): void {
 | |
|     let listOld: List = undefined;
 | |
|     try {
 | |
|         listOld = readJsonFile(filename) as List;
 | |
|     } catch (err) {
 | |
|         listOld = undefined;
 | |
|     }
 | |
|     if (listOld !== undefined) {
 | |
|         list.version = listOld.version; // take over
 | |
|         list.timestamp = listOld.timestamp; // take over
 | |
|         const diffs = diffTokenlist(list, listOld);
 | |
|         if (diffs != undefined) {
 | |
|             //console.log("List has Changed", JSON.stringify(diffs));
 | |
|             list.version = new Version(list.version.major + 1, 0, 0);
 | |
|             list.timestamp = (new Date()).toISOString();
 | |
|             console.log(`Version and timestamp updated, ${list.version.major}.${list.version.minor}.${list.version.patch} timestamp ${list.timestamp}`);
 | |
|         }
 | |
|     }
 | |
|     writeToFile(filename, list);
 | |
| }
 | |
| 
 | |
| async function addTokenIfNeeded(token: TokenItem, list: List): Promise<void> {
 | |
|     if (list.tokens.map(t => t.address.toLowerCase()).includes(token.address.toLowerCase())) {
 | |
|         return;
 | |
|     }
 | |
|     token = await updateTokenInfo(token);
 | |
|     list.tokens.push(token);
 | |
| }
 | |
| 
 | |
| // Update/fix token info, with properties retrieved from TW API
 | |
| async function updateTokenInfo(token: TokenItem): Promise<TokenItem> {
 | |
|     const tokenInfo: TokenTwInfo = await tokenInfoFromTwApi(token.asset);
 | |
|     if (tokenInfo) {
 | |
|         if (token.name && token.name != tokenInfo.name) {
 | |
|             console.log(`Token name adjusted: '${token.name}' -> '${tokenInfo.name}'`);
 | |
|             token.name = tokenInfo.name;
 | |
|         }
 | |
|         if (token.symbol && token.symbol != tokenInfo.symbol) {
 | |
|             console.log(`Token symbol adjusted: '${token.symbol}' -> '${tokenInfo.symbol}'`);
 | |
|             token.symbol = tokenInfo.symbol;
 | |
|         }
 | |
|         if (token.decimals && token.decimals != tokenInfo.decimals) {
 | |
|             console.log(`Token decimals adjusted: '${token.decimals}' -> '${tokenInfo.decimals}'`);
 | |
|             token.decimals = parseInt(tokenInfo.decimals.toString());
 | |
|         }
 | |
|     }
 | |
|     return token;
 | |
| }
 | |
| 
 | |
| function addPairToToken(pairToken: TokenItem, token: TokenItem, list: List): void {
 | |
|     const tokenInList = list.tokens.find(t => t.address === token.address);
 | |
|     if (!tokenInList) {
 | |
|         return;
 | |
|     }
 | |
|     if (!tokenInList.pairs) {
 | |
|         tokenInList.pairs = [];
 | |
|     }
 | |
|     if (tokenInList.pairs.map(p => p.base).includes(pairToken.asset)) {
 | |
|         return;
 | |
|     }
 | |
|     tokenInList.pairs.push(new Pair(pairToken.asset));
 | |
| }
 | |
| 
 | |
| function checkTokenExists(id: string, chainName: string, tokenAllowlist: string[]): boolean {
 | |
|     const logoPath = getChainAssetLogoPath(chainName, id);
 | |
|     if (!isPathExistsSync(logoPath)) {
 | |
|         //console.log("logo file missing", logoPath);
 | |
|         return false;
 | |
|     }
 | |
|     if (tokenAllowlist.find(t => (id.toLowerCase() === t.toLowerCase())) === undefined) {
 | |
|         //console.log(`Token not found in allowlist, ${id}`);
 | |
|         return false;
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| export async function addPairIfNeeded(token0: TokenItem, token1: TokenItem, list: List): Promise<void> {
 | |
|     await addTokenIfNeeded(token0, list);
 | |
|     await addTokenIfNeeded(token1, list);
 | |
|     addPairToToken(token1, token0, list);
 | |
|     // reverse direction not needed addPairToToken(token0, token1, list);
 | |
| }
 | |
| 
 | |
| function sort(list: List) {
 | |
|     list.tokens.sort((t1, t2) => {
 | |
|         const t1pairs = (t1.pairs || []).length;
 | |
|         const t2pairs = (t2.pairs || []).length;
 | |
|         if (t1pairs != t2pairs) { return t2pairs - t1pairs; }
 | |
|         return t1.address.localeCompare(t2.address);
 | |
|     });
 | |
|     list.tokens.forEach(t => {
 | |
|         t.pairs.sort((p1, p2) => p1.base.localeCompare(p2.base));
 | |
|     });
 | |
| }
 | |
| 
 | |
| function clearUnimportantFields(list: List) {
 | |
|     list.timestamp = "";
 | |
|     list.version = new Version(0, 0, 0);
 | |
| }
 | |
| 
 | |
| export function diffTokenlist(listOrig1: List, listOrig2: List): unknown {
 | |
|     // deep copy, to avoid changes
 | |
|     const list1 = JSON.parse(JSON.stringify(listOrig1));
 | |
|     const list2 = JSON.parse(JSON.stringify(listOrig2));
 | |
|     clearUnimportantFields(list1);
 | |
|     clearUnimportantFields(list2);
 | |
|     sort(list1);
 | |
|     sort(list2);
 | |
|     // compare
 | |
|     const diffs = diff(list1, list2);
 | |
|     return diffs;
 | |
| }
 | |
| 
 | |
| export async function rebuildTokenlist(chainName: string, pairs: [TokenItem, TokenItem][], listName: string): Promise<void> {
 | |
|     // sanity check, prevent deletion of many pairs
 | |
|     if (!pairs || pairs.length < 5) {
 | |
|         console.log(`Warning: Only ${pairs.length} pairs returned, ignoring`);
 | |
|         return;
 | |
|     }
 | |
|     
 | |
|     // filter out missing tokens
 | |
|     // prepare phase, read allowlist
 | |
|     const allowlist: string[] = readJsonFile(getChainAllowlistPath(chainName)) as string[];
 | |
|     const pairs2: [TokenItem, TokenItem][] = [];
 | |
|     pairs.forEach(p => {
 | |
|         if (!checkTokenExists(p[0].address, chainName, allowlist)) {
 | |
|             console.log("pair with unsupported 1st coin:", p[0].symbol, "--", p[1].symbol);
 | |
|             return;
 | |
|         }
 | |
|         if (!checkTokenExists(p[1].address, chainName, allowlist)) {
 | |
|             console.log("pair with unsupported 2nd coin:", p[0].symbol, "--", p[1].symbol);
 | |
|             return;
 | |
|         }
 | |
|         pairs2.push(p);
 | |
|     });
 | |
|     const filteredCount: number = pairs.length - pairs2.length;
 | |
|     console.log(`${filteredCount} unsupported tokens filtered out, ${pairs2.length} pairs`);
 | |
| 
 | |
|     const tokenlistFile = getChainTokenlistPath(chainName);
 | |
|     {
 | |
|         // show current size
 | |
|         const json = readJsonFile(tokenlistFile);
 | |
|         const list: List = json as List;
 | |
|         console.log(`Tokenlist original: ${list.tokens.length} tokens`);
 | |
|     }
 | |
|     const tokenlistBaseFile = getChainTokenlistBasePath(chainName);
 | |
|     const json = readJsonFile(tokenlistBaseFile);
 | |
|     const list: List = json as List;
 | |
|     console.log(`Tokenlist base: ${list.tokens.length} tokens`);
 | |
| 
 | |
|     await bluebird.each(pairs2, async (p) => {
 | |
|         await addPairIfNeeded(p[0], p[1], list);
 | |
|     });
 | |
|     console.log(`Tokenlist updated: ${list.tokens.length} tokens`);
 | |
| 
 | |
|     const newList = createTokensList(listName, list.tokens,
 | |
|         "2021-01-29T01:02:03.000+00:00", // use constant here to prevent changing time every time
 | |
|         0, 1, 0);
 | |
|     writeToFileWithUpdate(tokenlistFile, newList);
 | |
| }
 |