Неявное приведение не может работать с пробелами и недопустимыми символами в объектных ключах

В AlphaVantage API есть пробелы и точки в ключах. Для API нет официальной документации, хотя вы можете увидеть ее в их демонстрационном URL-адресе.

https://www.alphavantage.co/query?function=TIME_SERIES_INTRADAY&symbol=MSFT&interval=5min&apikey=demo

В моем приложении Typescript я создал для этого структуры данных (я рад, что кто-нибудь скопирует и использует их - возможно, после того, как будет найдено решение моего вопроса):

export class MetaData {
  '1. Information': string
  '2. Symbol': string
  '3. Last Refreshed': string
  '4. Output Size': string
  '5. Time Zone': string

  constructor(one, two, three, four, five) {
    this['1. Information'] = one
    this['2. Symbol'] = two
    this['3. Last Refreshed'] = three
    this['4. Output Size'] = four
    this['5. Time Zone'] = five
  }
}

export interface TimeSeries {
  [date: string]: {
    '1. open': string;
    '2. high': string;
    '3. low': string;
    '4. close': string;
    '5. volume': string;
  }
}

export interface AlphaVantage {
  'Meta Data': MetaData;
  'Time Series (Daily)'?: TimeSeries;
  'Time Series (Weekly)'?: TimeSeries;
}

Я вызываю API, используя alphavantage из NPM, и неявно передаю его своему AlphaVantage:

const av: AlphaVantage = await alpha.data.weekly(options.stocks, 'compact', 'json')

И затем (возможно, после некоторого массажа и т. Д.) Я сохраняю его в коллекции MongoDB:

const doc = await this.model.findByIdAndUpdate(proxyModel._id, proxyModel)

(ProxyModel - это DTO, используемый для определения ключей базы данных, таких как дата, символ акций и т. Д. ... Одно из полей - данные AlphaVantage).

Это должно сериализовать данные и ошибки:

key 1. Information must not contain '.'

Есть простой способ справиться с этим. Я бы предпочел создать эквивалентные объекты без пробелов:

export interface TimeSeries {
  [date: string]: {
    '1_open': string;
    '2_high': string;
    '3_low': string;
    '4_close': string;
    '5_volume': string;
  }
}

А затем бросить на это. В этом случае предоставьте отображение ...

Я вижу, как я создаю реализацию. Однако прежде чем я перейду к этому, я хотел бы услышать какие-либо идеи, как лучше всего обрабатывать эту структуру данных.


person HankCa    schedule 15.05.2019    source источник


Ответы (1)


Я написал решение с простой функцией сопоставления клавиш объекта.

Я пробовал также использовать AutoMapper-ts - https://www.npmjs.com/package/automapper-ts - чтобы было понятнее о вносимых изменениях. Однако отображать все случаи стало слишком сложно. Я не мог заставить тесты (которые действуют как doco) работать в то время, которое я на это выделил. Я только что видел, что есть https://github.com/gonza-lito/AutoMapper, который является недавно модифицированной вилкой. Однако это не npm installing из коробки.

Это класс, который я придумал для решения своей проблемы:

export class ObjectUtils {

  static fixKey(key) {
    const upperCaseReplacer = (match: string, g1: string) => {
      return g1.toUpperCase()
    }
    const wordsReplacer = (match: string, g1: string, g2: string) => {
      return g1 + g2.toUpperCase()
    }
    const m1 = key.replace(/^\d+ *\. *([a-zA-Z])/, upperCaseReplacer)
    const m2 = m1.replace(/([a-zA-Z]) +([a-zA-Z])/g, wordsReplacer)
    const out = m2.replace(/^([a-zA-Z])/, upperCaseReplacer)
    return out
  }

  static fixObj(obj: any) {
    let newObj = null
    if (Array.isArray(obj)) {
      newObj = []
      for (const i of obj) {
        newObj.push(ObjectUtils.fixObj(i))
      }
      return newObj
    } else if (typeof obj === 'object') {
      newObj = {}
      for (const key of Object.keys(obj)) {
        newObj[ObjectUtils.fixKey(key)] = ObjectUtils.fixObj(obj[key])
      }
      return newObj
    } else {
      return obj
    }
  }
}

Это создает следующее:

export class Metadata {
  Information: string
  Symbol: string
  LastRefreshed: string
  OutputSize: string
  TimeZone: string
}

export interface TimeSeries {
  [date: string]: {
    Open: string;
    High: string;
    Low: string;
    Close: string;
    Volume: string;
  }
}

export interface AlphaVantage {
  Metadata: Metadata
  DailyTimeSeries?: TimeSeries
  WeeklyTimeSeries?: TimeSeries
}
person HankCa    schedule 19.05.2019