diff --git a/.github/workflows/botonic-plugin-contentful-tests.yml b/.github/workflows/botonic-plugin-contentful-tests.yml index 4b2e674f45..c4e2ec839f 100644 --- a/.github/workflows/botonic-plugin-contentful-tests.yml +++ b/.github/workflows/botonic-plugin-contentful-tests.yml @@ -25,7 +25,7 @@ jobs: - name: Install dev dependencies run: (cd ./packages/$PACKAGE && npm install -D) - name: Build - run: (cd ./packages/$PACKAGE && npm run build) + run: (cd ./packages/$PACKAGE && npm run build_with_tests) - name: Run tests env: CONTENTFUL_TEST_SPACE_ID: ${{ secrets.CONTENTFUL_TEST_SPACE_ID }} diff --git a/.github/workflows/botonic-plugin-dynamo-tests.yml b/.github/workflows/botonic-plugin-dynamo-tests.yml index c32e86ce16..577656fde6 100644 --- a/.github/workflows/botonic-plugin-dynamo-tests.yml +++ b/.github/workflows/botonic-plugin-dynamo-tests.yml @@ -25,7 +25,7 @@ jobs: - name: Install dev dependencies run: (cd ./packages/$PACKAGE && npm install -D) - name: Build - run: (cd ./packages/$PACKAGE && npm run build) + run: (cd ./packages/$PACKAGE && npm run build_with_tests) - name: Run tests env: AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} diff --git a/package-lock.json b/package-lock.json index 14d4bca6fc..4d0eec22e6 100644 --- a/package-lock.json +++ b/package-lock.json @@ -10898,9 +10898,9 @@ } }, "typescript": { - "version": "3.7.5", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.7.5.tgz", - "integrity": "sha512-/P5lkRXkWHNAbcJIiHPfRoKqyd7bsyCma1hZNUGfn20qm64T6ZBlrzprymeu918H+mB/0rIg2gGK/BXkhhYgBw==", + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.8.3.tgz", + "integrity": "sha512-MYlEfn5VrLNsgudQTVJeNaQFUAI7DkhnOjdpAp4T+ku1TfQClewlbSuTVHiA+8skNBgaf02TL/kLOvig4y3G8w==", "dev": true }, "uglify-js": { diff --git a/package.json b/package.json index f9e1c5fba3..a1f18270d5 100644 --- a/package.json +++ b/package.json @@ -22,7 +22,7 @@ "ts-mockito": "^2.5.0", "ts-node": "^8.6.2", "tslib": "^1.10.0", - "typescript": "~3.7.5" + "typescript": "^3.8.3" }, "engines": { "node": ">=8.0.0" diff --git a/packages/botonic-plugin-contentful/jest.setup.js b/packages/botonic-plugin-contentful/jest.setup.js index 4fd514ad16..96e13e6814 100644 --- a/packages/botonic-plugin-contentful/jest.setup.js +++ b/packages/botonic-plugin-contentful/jest.setup.js @@ -1,2 +1,2 @@ // uncomment to extend jest test timeout while debugging from IDE -// jest.setTimeout(300000); +// jest.setTimeout(3000000) diff --git a/packages/botonic-plugin-contentful/package-lock.json b/packages/botonic-plugin-contentful/package-lock.json index 34fe67004b..51e8d3facf 100644 --- a/packages/botonic-plugin-contentful/package-lock.json +++ b/packages/botonic-plugin-contentful/package-lock.json @@ -1054,6 +1054,11 @@ "printj": "~1.1.0" } }, + "csv-parse": { + "version": "4.8.6", + "resolved": "https://registry.npmjs.org/csv-parse/-/csv-parse-4.8.6.tgz", + "integrity": "sha512-rSJlpgAjrB6pmlPaqiBAp3qVtQHN07VxI+ozs+knMsNvgh4bDQgENKLFYLFMvT+jn/wr/zvqsd7IVZ7Txdkr7w==" + }, "csv-stringify": { "version": "5.3.6", "resolved": "https://registry.npmjs.org/csv-stringify/-/csv-stringify-5.3.6.tgz", @@ -2365,6 +2370,14 @@ } } }, + "sort-stream": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/sort-stream/-/sort-stream-1.0.1.tgz", + "integrity": "sha1-owaEycKcozMGnBjWoKsPdor2HfI=", + "requires": { + "through": "~2.3.1" + } + }, "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", @@ -2442,6 +2455,11 @@ "has-flag": "^3.0.0" } }, + "through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=" + }, "timers-ext": { "version": "0.1.7", "resolved": "https://registry.npmjs.org/timers-ext/-/timers-ext-0.1.7.tgz", diff --git a/packages/botonic-plugin-contentful/package.json b/packages/botonic-plugin-contentful/package.json index c512df4b02..8dc572ec2a 100644 --- a/packages/botonic-plugin-contentful/package.json +++ b/packages/botonic-plugin-contentful/package.json @@ -25,11 +25,14 @@ "dependencies": { "@babel/runtime": "^7.8.3", "contentful": "^7.13.1", + "csv-parse": "^4.8.6", + "csv-stringify": "^5.3.6", "escape-string-regexp": "^2.0.0", "memoizee": "^0.4.14", "moment": "^2.24.0", "moment-timezone": "^0.5.27", - "node-nlp": "^4.0.2" + "node-nlp": "^4.0.2", + "sort-stream": "^1.0.1" }, "devDependencies": { "@types/memoizee": "^0.4.3", diff --git a/packages/botonic-plugin-contentful/src/cms/callback.ts b/packages/botonic-plugin-contentful/src/cms/callback.ts index 1b452962e0..ecc34f93b0 100644 --- a/packages/botonic-plugin-contentful/src/cms/callback.ts +++ b/packages/botonic-plugin-contentful/src/cms/callback.ts @@ -1,4 +1,9 @@ -import { CMS, ContentType, MESSAGE_TYPES, MessageContentType } from './cms' +import { + CMS, + ContentType, + MESSAGE_CONTENT_TYPES, + MessageContentType, +} from './cms' import escapeStringRegexp from 'escape-string-regexp' import { Context } from './context' import { TopContent } from './contents' @@ -54,7 +59,7 @@ export class ContentCallback extends Callback { } private static checkDeliverableModel(modelType: string): MessageContentType { - if (MESSAGE_TYPES.includes(modelType as MessageContentType)) { + if (MESSAGE_CONTENT_TYPES.includes(modelType as MessageContentType)) { return modelType as MessageContentType } else { throw new Error( diff --git a/packages/botonic-plugin-contentful/src/cms/cms.ts b/packages/botonic-plugin-contentful/src/cms/cms.ts index 2819acbbdb..87cb4938a0 100644 --- a/packages/botonic-plugin-contentful/src/cms/cms.ts +++ b/packages/botonic-plugin-contentful/src/cms/cms.ts @@ -15,6 +15,7 @@ import { TopContent, Content, } from './contents' +import { enumValues } from '../util/enums' export enum MessageContentType { CAROUSEL = 'carousel', @@ -23,8 +24,10 @@ export enum MessageContentType { CHITCHAT = 'chitchat', //so far it's an alias for TEXT STARTUP = 'startUp', } -export const MESSAGE_TYPES = Object.values(MessageContentType).map( - m => m as MessageContentType + +// CHITCHAT removed because it's an alias for texts +export const MESSAGE_CONTENT_TYPES = enumValues(MessageContentType).filter( + m => m != MessageContentType.CHITCHAT ) export enum NonMessageTopContentType { @@ -39,9 +42,10 @@ export const TopContentType = { ...MessageContentType, ...NonMessageTopContentType, } -export const TOPCONTENT_TYPES = Object.values(TopContentType).map( - m => m as TopContentType -) +export const TOP_CONTENT_TYPES = [ + ...MESSAGE_CONTENT_TYPES, + ...enumValues(NonMessageTopContentType), +] export enum SubContentType { BUTTON = 'button', @@ -49,9 +53,17 @@ export enum SubContentType { } export type ContentType = TopContentType | SubContentType export const ContentType = { ...TopContentType, ...SubContentType } -export const CONTENT_TYPES = Object.values(ContentType).map( - m => m as ContentType -) +export const CONTENT_TYPES = [ + ...TOP_CONTENT_TYPES, + ...enumValues(SubContentType), +] + +export type BotonicContentType = MessageContentType | SubContentType +export const BotonicContentType = { ...MessageContentType, ...SubContentType } +export const BOTONIC_CONTENT_TYPES = [ + ...MESSAGE_CONTENT_TYPES, + ...enumValues(SubContentType), +] export function isSameModel(model1: ContentType, model2: ContentType): boolean { switch (model1) { diff --git a/packages/botonic-plugin-contentful/src/cms/context.ts b/packages/botonic-plugin-contentful/src/cms/context.ts index 9b853ef204..bbf5ed9d88 100644 --- a/packages/botonic-plugin-contentful/src/cms/context.ts +++ b/packages/botonic-plugin-contentful/src/cms/context.ts @@ -9,4 +9,9 @@ export const DEFAULT_CONTEXT: Context = {} export interface Context { locale?: Locale callbacks?: CallbackMap + /** + * When set, empty fields will be blank even if they have a value for the fallback locale + * NOT applying it so far for assets because cms.Asset does not support blank assets + */ + ignoreFallbackLocale?: boolean } diff --git a/packages/botonic-plugin-contentful/src/contentful/contents/button.ts b/packages/botonic-plugin-contentful/src/contentful/contents/button.ts index a1e76e8372..7f623e074f 100644 --- a/packages/botonic-plugin-contentful/src/contentful/contents/button.ts +++ b/packages/botonic-plugin-contentful/src/contentful/contents/button.ts @@ -12,7 +12,7 @@ import { TextFields } from './text' import { UrlFields } from './url' export class ButtonDelivery { - private static BUTTON_CONTENT_TYPE = 'button' + public static BUTTON_CONTENT_TYPE = 'button' private static PAYLOAD_CONTENT_TYPE = 'payload' constructor(private readonly delivery: DeliveryApi) {} diff --git a/packages/botonic-plugin-contentful/src/contentful/contents/carousel.ts b/packages/botonic-plugin-contentful/src/contentful/contents/carousel.ts index ce052f72f3..64d63eb898 100644 --- a/packages/botonic-plugin-contentful/src/contentful/contents/carousel.ts +++ b/packages/botonic-plugin-contentful/src/contentful/contents/carousel.ts @@ -36,7 +36,7 @@ export class CarouselDelivery extends DeliveryWithFollowUp { ) } - private async elementFromEntry( + public async elementFromEntry( entry: contentful.Entry, context: cms.Context ): Promise { diff --git a/packages/botonic-plugin-contentful/src/contentful/delivery-api.ts b/packages/botonic-plugin-contentful/src/contentful/delivery-api.ts index a902e8d221..73de8fe52b 100644 --- a/packages/botonic-plugin-contentful/src/contentful/delivery-api.ts +++ b/packages/botonic-plugin-contentful/src/contentful/delivery-api.ts @@ -35,6 +35,8 @@ export interface DeliveryApi { context: Context, query?: any ): Promise> + + getContentType(id: string): Promise } /** @@ -67,6 +69,15 @@ export class AdaptorDeliveryApi implements DeliveryApi { ) } + async getContentType(id: string): Promise { + try { + return this.client.getContentType(id) + } catch (e) { + console.error(`ERROR in getContentType for id ${id}: ${e}`) + throw e + } + } + private static queryFromContext(context: Context, query: any = {}): any { if (context.locale) { query['locale'] = context.locale diff --git a/packages/botonic-plugin-contentful/src/contentful/delivery/cache.ts b/packages/botonic-plugin-contentful/src/contentful/delivery/cache.ts index 461c0f26df..0a1eaa0782 100644 --- a/packages/botonic-plugin-contentful/src/contentful/delivery/cache.ts +++ b/packages/botonic-plugin-contentful/src/contentful/delivery/cache.ts @@ -1,5 +1,5 @@ import * as contentful from 'contentful/index' -import { Entry } from 'contentful/index' +import { ContentType, Entry } from 'contentful/index' import memoize from 'memoizee' import { ReducedClientApi } from './client-api' @@ -7,6 +7,7 @@ export class CachedClientApi implements ReducedClientApi { readonly getAsset: (id: string, query?: any) => Promise readonly getEntries: (query: any) => Promise> readonly getEntry: (id: string, query?: any) => Promise> + readonly getContentType: (id: string) => Promise constructor(readonly client: ReducedClientApi, readonly cacheTtlMs = 10000) { const options = (length: number) => @@ -24,5 +25,6 @@ export class CachedClientApi implements ReducedClientApi { this.getAsset = memoize(client.getAsset, options(2)) this.getEntries = memoize(client.getEntries, options(1)) this.getEntry = memoize(client.getEntry, options(2)) + this.getContentType = memoize(client.getContentType, options(1)) } } diff --git a/packages/botonic-plugin-contentful/src/contentful/delivery/client-api.ts b/packages/botonic-plugin-contentful/src/contentful/delivery/client-api.ts index d169f904bb..6c79d72262 100644 --- a/packages/botonic-plugin-contentful/src/contentful/delivery/client-api.ts +++ b/packages/botonic-plugin-contentful/src/contentful/delivery/client-api.ts @@ -2,5 +2,5 @@ import { ContentfulClientApi } from 'contentful' export type ReducedClientApi = Pick< ContentfulClientApi, - 'getAsset' | 'getEntries' | 'getEntry' + 'getAsset' | 'getEntries' | 'getEntry' | 'getContentType' > diff --git a/packages/botonic-plugin-contentful/src/contentful/ignore-fallback-decorator.ts b/packages/botonic-plugin-contentful/src/contentful/ignore-fallback-decorator.ts new file mode 100644 index 0000000000..e297e06ea8 --- /dev/null +++ b/packages/botonic-plugin-contentful/src/contentful/ignore-fallback-decorator.ts @@ -0,0 +1,118 @@ +import { Asset, ContentType, Entry, EntryCollection } from 'contentful' +import { DeliveryApi } from './delivery-api' +import { Context } from '../cms' +import { + ContentfulVisitor, + I18nEntryTraverser, + I18nValue, + VisitedField, +} from './traverser' + +export class IgnoreFallbackDecorator implements DeliveryApi { + constructor(private readonly api: DeliveryApi) {} + + getContentType(id: string): Promise { + return this.api.getContentType(id) + } + + async getEntries( + context: Context, + query: any = {} + ): Promise> { + if (!context.ignoreFallbackLocale) { + return this.api.getEntries(context, query) + } + let entries = await this.api.getEntries(this.i18nContext(context), query) + + entries = { ...entries } + entries.items = await this.traverseEntries(context, entries.items) + return entries + } + + async getEntry( + id: string, + context: Context, + query: any = {} + ): Promise> { + if (!context.ignoreFallbackLocale) { + return this.api.getEntry(id, context, query) + } + const entry = await this.api.getEntry( + id, + this.i18nContext(context), + query + ) + return (await this.traverseEntries(context, [entry]))[0] + } + + async traverseEntries( + context: Context, + entries: Entry[] + ): Promise[]> { + const visitor = new IgnoreFallbackVisitor(context) + return Promise.all( + entries.map(async item => { + const traverser = new I18nEntryTraverser(this.api, visitor) + return await traverser.traverse(item, context) + }) + ) + } + + getAsset(id: string, query?: any): Promise { + return this.api.getAsset(id, query) + } + + private i18nContext(context: Context) { + return { + ...context, + locale: '*', + } as Context + } +} + +class IgnoreFallbackVisitor implements ContentfulVisitor { + contextForContentful: Context + constructor(readonly context: Context) { + if (!context.locale) { + throw new Error( + 'Context.ignoreFallbackLocale set but Context.locale not set' + ) + } + this.contextForContentful = { + ...context, + locale: '*', + } + } + + visitEntry(entry: Entry): Entry { + return entry + } + + visitStringField(vf: VisitedField): I18nValue { + return this.hackType(vf.value[vf.locale], '') + } + hackType(t: T, defaultValue?: T): I18nValue { + if (defaultValue != undefined) { + t = t ?? defaultValue + } + return (t as any) as I18nValue + } + + visitMultipleStringField(vf: VisitedField): I18nValue { + return this.hackType(vf.value[vf.locale], []) + } + + visitSingleReference(vf: VisitedField>): I18nValue> { + return this.hackType(vf.value[vf.locale], (undefined as any) as Entry) + } + + visitMultipleReference( + vf: VisitedField> + ): I18nValue> { + return this.hackType(vf.value[vf.locale]) + } + + name(): string { + return 'ignoreFallbackLocale' + } +} diff --git a/packages/botonic-plugin-contentful/src/contentful/index.ts b/packages/botonic-plugin-contentful/src/contentful/index.ts index 5b84c75c45..e82658ba82 100644 --- a/packages/botonic-plugin-contentful/src/contentful/index.ts +++ b/packages/botonic-plugin-contentful/src/contentful/index.ts @@ -32,6 +32,7 @@ import * as contentful from 'contentful' import { ContentfulOptions } from '../plugin' import { CachedClientApi } from './delivery/cache' import { CreateClientParams } from 'contentful' +import { IgnoreFallbackDecorator } from './ignore-fallback-decorator' export default class Contentful implements cms.CMS { private readonly _delivery: DeliveryApi @@ -56,21 +57,13 @@ export default class Contentful implements cms.CMS { * https://www.contentful.com/developers/docs/javascript/tutorials/using-js-cda-sdk/ */ constructor(options: ContentfulOptions) { - const params: CreateClientParams = { - space: options.spaceId, - accessToken: options.accessToken, - timeout: options.timeoutMs, - } - if (options.environment) { - params.environment = options.environment - } - const client = contentful.createClient(params) + const client = createContentfulClientApi(options) const deliveryApi = new AdaptorDeliveryApi( options.disableCache ? client : new CachedClientApi(client, options.cacheTtlMs) ) - const delivery = deliveryApi + const delivery = new IgnoreFallbackDecorator(deliveryApi) this._contents = new ContentsApi(delivery) this._delivery = delivery @@ -134,7 +127,6 @@ export default class Contentful implements cms.CMS { context = DEFAULT_CONTEXT, filter?: (cf: CommonFields) => boolean ): Promise { - console.log('getting contents for lang', context.locale) return this._contents.topContents( model, context, @@ -161,22 +153,27 @@ export default class Contentful implements cms.CMS { context: Context ): Promise { const model = ContentfulEntryUtils.getContentModel(entry) - switch (model) { - case ContentType.CAROUSEL: - return this._carousel.fromEntry(entry, context) - case ContentType.QUEUE: - return QueueDelivery.fromEntry(entry) - case ContentType.CHITCHAT: - case ContentType.TEXT: - return this._text.fromEntry(entry, context) - case ContentType.IMAGE: - return this._image.fromEntry(entry, context) - case ContentType.URL: - return this._url.fromEntry(entry, context) - case ContentType.STARTUP: - return this._startUp.fromEntry(entry, context) - default: - throw new Error(`${model} is not a Content type`) + try { + switch (model) { + case ContentType.CAROUSEL: + return await this._carousel.fromEntry(entry, context) + case ContentType.QUEUE: + return await QueueDelivery.fromEntry(entry) + case ContentType.CHITCHAT: + case ContentType.TEXT: + return await this._text.fromEntry(entry, context) + case ContentType.IMAGE: + return await this._image.fromEntry(entry, context) + case ContentType.URL: + return await this._url.fromEntry(entry, context) + case ContentType.STARTUP: + return await this._startUp.fromEntry(entry, context) + default: + throw new Error(`${model} is not a Content type`) + } + } catch (e) { + console.error(`Error creating ${model} with id: ${entry.sys.id}`) + throw e } } @@ -188,6 +185,8 @@ export default class Contentful implements cms.CMS { switch (model) { case ContentType.BUTTON: return this._button.fromEntry(entry, context) + case ContentType.ELEMENT: + return this._carousel.elementFromEntry(entry, context) default: return this.topContentFromEntry(entry, context) } @@ -212,5 +211,20 @@ export default class Contentful implements cms.CMS { } } +export function createContentfulClientApi( + options: ContentfulOptions +): contentful.ContentfulClientApi { + const params: CreateClientParams = { + space: options.spaceId, + accessToken: options.accessToken, + timeout: options.timeoutMs, + } + if (options.environment) { + params.environment = options.environment + } + const client = contentful.createClient(params) + return client +} + export { DeliveryApi } from './delivery-api' export { CarouselDelivery } from './contents/carousel' diff --git a/packages/botonic-plugin-contentful/src/contentful/traverser.ts b/packages/botonic-plugin-contentful/src/contentful/traverser.ts new file mode 100644 index 0000000000..626e413d92 --- /dev/null +++ b/packages/botonic-plugin-contentful/src/contentful/traverser.ts @@ -0,0 +1,209 @@ +import * as cf from 'contentful' + +import { Context } from '../cms' +import { DeliveryApi } from './delivery-api' +import { Locale } from '../nlp' +import { ButtonDelivery } from './contents/button' + +export type I18nValue = { [locale: string]: T } + +export class VisitedField { + constructor( + readonly entry: cf.Entry, + readonly locale: Locale, + readonly field: cf.Field, + readonly value: I18nValue + ) {} +} +export interface ContentfulVisitor { + name(): string + + visitEntry(entry: cf.Entry): cf.Entry + + visitStringField(field: VisitedField): I18nValue + + visitMultipleStringField(field: VisitedField): I18nValue + + visitSingleReference( + field: VisitedField> + ): I18nValue> + + visitMultipleReference( + field: VisitedField> + ): I18nValue> +} + +export class LoggerContentfulVisitor implements ContentfulVisitor { + constructor(private readonly visitor: ContentfulVisitor) {} + name(): string { + return this.visitor.name() + } + visitEntry(entry: cf.Entry): cf.Entry { + this.log('visitEntry', entry) + return this.visitor.visitEntry(entry) + } + + visitStringField(field: VisitedField): I18nValue { + this.log('visitStringField', field.entry, field.field) + return this.visitor.visitStringField(field) + } + + visitMultipleStringField( + field: VisitedField + ): I18nValue { + this.log('visitMultipleStringField', field.entry, field.field) + return this.visitor.visitMultipleStringField(field) + } + + visitSingleReference( + field: VisitedField> + ): I18nValue> { + this.log('visitSingleReference', field.entry, field.field) + return this.visitor.visitSingleReference(field) + } + + visitMultipleReference( + field: VisitedField> + ): I18nValue> { + this.log('visitMultipleReference', field.entry, field.field) + return this.visitor.visitMultipleReference(field) + } + + log(method: string, entry: cf.Entry, field?: cf.Field): void { + const on = field + ? LoggerContentfulVisitor.describeField(entry, field.id) + : LoggerContentfulVisitor.describeEntry(entry) + console.log(`Visiting '${this.visitor.name()}.${method}' on ${on}`) + } + + static describeEntry(entry: cf.Entry): string { + if (!entry.sys.contentType) { + return `entry with id ${entry.sys.id}` + } + return `entry of type ${entry.sys.contentType.sys.id} (id:${entry.sys.id})` + } + + static describeField(entry: cf.Entry, name: string): string { + // cannot stringify field values because they may contain circular references + return `field '${name}' of ${LoggerContentfulVisitor.describeEntry(entry)}` + } +} + +/** + * Traverser a contentful Entry which has been requested for all locales. + * Limitations. It does not fetch entries from references which have not yet been delivered + */ +export class I18nEntryTraverser { + private visited = new Set() + constructor( + private readonly api: DeliveryApi, + readonly visitor: ContentfulVisitor + ) {} + + async traverse( + entry: cf.Entry, + context: Context + ): Promise> { + //in the future we might extending to traverse all locales + console.assert(context.locale) + console.assert(context.ignoreFallbackLocale) + const promise = this.traverseCore(entry, context) + this.visited.add(entry.sys.id) + return promise + } + + async traverseCore( + entry: cf.Entry, + context: Context + ): Promise> { + entry = { ...entry, fields: { ...entry.fields } } + const fields = (entry.fields as unknown) as { + [fieldName: string]: I18nValue + } + if (!entry.sys.contentType) { + // it's a file or a dangling reference + return entry + } + const contentType = await this.api.getContentType( + entry.sys.contentType.sys.id + ) + for (const fieldId in fields) { + const field = contentType.fields.find(f => f.id == fieldId)! + const i18nValue = { ...fields[fieldId] } as I18nValue + const locale = field.localized + ? context.locale! + : Object.keys(i18nValue)[0] + const vf = new VisitedField(entry, locale, field, i18nValue) + fields[fieldId] = await this.traverseField(context, vf) + } + entry = this.visitor.visitEntry(entry) + return entry + } + + async traverseField( + context: Context, + vf: VisitedField + ): Promise> { + let val = vf.value[vf.locale] + + const visitOrTraverse = async (val: cf.Entry) => { + if (this.visited.has(val.sys.id) && val.sys.id >= vf.entry.sys.id) { + // break deadlock if contents have cyclic dependencies + return val + } + return this.traverse(val, context) + } + if (vf.field.type === 'Symbol' || vf.field.type === 'Text') { + return this.visitor.visitStringField(vf) + } else if (vf.field.type == 'Link') { + if (val) { + val = this.stopRecursionOnButtonCallbacks(vf.field, val) + vf.value[vf.locale] = visitOrTraverse(val) + } + return this.visitor.visitSingleReference(vf) + } else if (this.isArrayOfType(vf.field, 'Link')) { + if (val) { + val = await Promise.all( + (val as cf.Entry[]).map(v => visitOrTraverse(v)) + ) + vf.value[vf.locale] = val + } + return this.visitor.visitMultipleReference(vf) + } else if (this.isArrayOfType(vf.field, 'Symbol')) { + return this.visitor.visitMultipleStringField(vf) + } else { + console.log( + `Not traversing ${LoggerContentfulVisitor.describeField( + vf.entry, + vf.field.id + )}'}` + ) + return vf.value + } + } + + isArrayOfType(field: cf.Field, itemType: cf.FieldType): boolean { + return field.type == 'Array' && field.items?.type == itemType + } + + /** + * When a content has a button with another content reference, we just need the referered content id + * to create the content. Hence, we stop traversing. + */ + private stopRecursionOnButtonCallbacks( + field: cf.Field, + val: cf.Entry + ): cf.Entry { + if (field.id !== 'target') { + return val + } + if ( + !val.fields || + val.fields.payload || + val.sys.contentType.sys.id == ButtonDelivery.BUTTON_CONTENT_TYPE + ) { + return val + } + return { ...val, fields: {} } + } +} diff --git a/packages/botonic-plugin-contentful/src/tools/translators/csv-export.ts b/packages/botonic-plugin-contentful/src/tools/translators/csv-export.ts new file mode 100644 index 0000000000..79c9f63d98 --- /dev/null +++ b/packages/botonic-plugin-contentful/src/tools/translators/csv-export.ts @@ -0,0 +1,139 @@ +import { + BOTONIC_CONTENT_TYPES, + Button, + CMS, + CommonFields, + Content, + Element, + StartUp, + TopContent, +} from '../../index' +import { Locale } from '../../nlp' +import { Text } from '../../cms' +import stringify from 'csv-stringify' +import * as stream from 'stream' +import * as fs from 'fs' +import { promisify } from 'util' +import sort from 'sort-stream' + +const finished = promisify(stream.finished) + +class I18nField { + constructor(readonly name: string, readonly value: string) {} +} + +type CsvLine = string[] + +interface CsvExportOptions { + readonly nameFilter?: (name: string) => boolean + readonly stringFilter?: (text: string) => boolean +} + +export const skipEmptyStrings = (str: string) => Boolean(str && str.trim()) +/*** + * Uses https://csv.js.org/stringify/api/ + */ +export class CsvExport { + private toFields: ContentToCsvLines + + constructor(private readonly options: CsvExportOptions) { + this.toFields = new ContentToCsvLines(options) + } + + create_stringifier() { + return stringify({ + escape: '"', + delimiter: ';', + quote: '"', + quoted: true, + record_delimiter: 'windows', + header: true, + columns: ['Model', 'Code', 'Id', 'Field', 'From', 'To'], + }) + } + + static sortRows(a: string[], b: string[]): number { + for (const i in a) { + const cmp = a[i].localeCompare(b[i]) + if (cmp != 0) { + return cmp + } + } + return 0 + } + + async write(fname: string, cms: CMS, locale: Locale): Promise { + const stringifier = this.create_stringifier() + const readable = stream.Readable.from(this.generate(cms, locale)) + const writable = readable + .pipe(sort(CsvExport.sortRows)) + .pipe(stringifier) + .pipe(fs.createWriteStream(fname)) + return this.toPromise(writable) + } + + async toPromise(writable: stream.Writable): Promise { + return finished(writable) + } + + async *generate(cms: CMS, from: Locale): AsyncGenerator { + for (const model of BOTONIC_CONTENT_TYPES) { + console.log(`Exporting contents of type ${model}`) + const contents = await cms.contents(model, { + locale: from, + ignoreFallbackLocale: true, + }) + for (const content of contents) { + if (this.options.nameFilter && !this.options.nameFilter(content.name)) { + continue + } + console.log('Exporting content', content.name.trim() || content.id) + for (const field of this.toFields.getCsvLines(content)) { + const TO_COLUMN = '' + yield [...field, TO_COLUMN] + } + } + } + } +} + +export class ContentToCsvLines { + constructor(private readonly options: CsvExportOptions) {} + + getCsvLines(content: Content): CsvLine[] { + const columns = [content.contentType, content.name, content.id] + let fields = this.getFields(content) + if (this.options.stringFilter) { + fields = fields.filter(f => this.options.stringFilter!(f.value)) + } + return fields.map(f => [...columns, f.name, f.value!]) + } + + getFields(content: Content): I18nField[] { + if (content instanceof Button) { + return [new I18nField('Text', content.text)] + } else if (content instanceof StartUp) { + return [ + ...this.getCommonFields(content.common), + new I18nField('Text', content.text), + ] + } else if (content instanceof Text) { + return [ + ...this.getCommonFields(content.common), + new I18nField('Text', content.text), + ] + } else if (content instanceof Element) { + return [ + new I18nField('Title', content.title), + new I18nField('Subtitle', content.subtitle), + ] + } else if (content instanceof TopContent) { + return this.getCommonFields(content.common) + } + return [] + } + + getCommonFields(common: CommonFields): I18nField[] { + return [new I18nField('Short text', common.shortText)] + } +} diff --git a/packages/botonic-plugin-contentful/src/tools/translators/csv-for-translator.ts b/packages/botonic-plugin-contentful/src/tools/translators/csv-for-translator.ts new file mode 100644 index 0000000000..2e2a013f77 --- /dev/null +++ b/packages/botonic-plugin-contentful/src/tools/translators/csv-for-translator.ts @@ -0,0 +1,45 @@ +import { CsvExport, skipEmptyStrings } from './csv-export' +import { Locale } from '../../nlp' +import Contentful from '../../contentful' + +async function writeCsvForTranslators( + spaceId: string, + accessToken: string, + locales: Locale[] +) { + const cms = new Contentful({ + spaceId: spaceId, + accessToken: accessToken, + environment: 'master', + }) + const exporter = new CsvExport({ + stringFilter: skipEmptyStrings, + }) + const promises = locales.map((from: string) => + exporter.write(`contentful_${from}.csv`, cms, from) + ) + await Promise.all(promises) +} + +const spaceId = process.argv[2] +const token = process.argv[3] +const locales = process.argv.slice(4) as Locale[] +console.log(process.execArgv) +if (process.argv.length < 5) { + console.error(`Usage: space_id access_token language`) + process.exit(1) +} + +async function main() { + try { + await writeCsvForTranslators(spaceId, token, locales) + console.log('done') + } catch (e) { + console.error(e) + } +} + +// eslint-disable-next-line @typescript-eslint/no-floating-promises +main().then(() => { + console.log('done') +}) diff --git a/packages/botonic-plugin-contentful/src/tools/translators/csv-import.ts b/packages/botonic-plugin-contentful/src/tools/translators/csv-import.ts new file mode 100644 index 0000000000..e69de29bb2 diff --git a/packages/botonic-plugin-contentful/src/typings.d.ts b/packages/botonic-plugin-contentful/src/typings.d.ts index e1cf10f3ad..5b9f384e08 100644 --- a/packages/botonic-plugin-contentful/src/typings.d.ts +++ b/packages/botonic-plugin-contentful/src/typings.d.ts @@ -74,3 +74,8 @@ declare module '@nlpjs/lang-ca/src' { export class StemmerCa extends BaseStemmer {} } + +declare module 'sort-stream' { + function sort(func: (a: any, b: any) => number): any + export = sort +} diff --git a/packages/botonic-plugin-contentful/src/util/enums.ts b/packages/botonic-plugin-contentful/src/util/enums.ts new file mode 100644 index 0000000000..97714fc4f5 --- /dev/null +++ b/packages/botonic-plugin-contentful/src/util/enums.ts @@ -0,0 +1,3 @@ +export function enumValues(enumType: { [s: string]: T }): T[] { + return Object.values(enumType).map(m => m as T) +} diff --git a/packages/botonic-plugin-contentful/src/util/objects.ts b/packages/botonic-plugin-contentful/src/util/objects.ts index fea5ed280e..3641157cb2 100644 --- a/packages/botonic-plugin-contentful/src/util/objects.ts +++ b/packages/botonic-plugin-contentful/src/util/objects.ts @@ -13,4 +13,41 @@ export function shallowClone(obj: T): T { return clone as T } -// consider this deepClone in TS https://gist.github.com/erikvullings/ada7af09925082cbb89f40ed962d475e +/** + * Deep copy function for TypeScript. + * @param T Generic type of target/copied value. + * @param target Target value to be copied. + * @see Source project, ts-deepcopy https://github.com/ykdr2017/ts-deepcopy + * @see Code pen https://codepen.io/erikvullings/pen/ejyBYg + */ +export const deepClone = (target: T, alreadyCloned: object[] = []): T => { + // @ts-ignore + if (alreadyCloned.includes(target)) { + return target + } + // @ts-ignore + alreadyCloned.push(target) + if (target === undefined) { + return target + } + if (target instanceof Date) { + return new Date(target.getTime()) as any + } + if (target instanceof Array) { + const cp = [] as any[] + ;(target as any[]).forEach(v => { + cp.push(v) + }) + return cp.map((n: any) => deepClone(n, alreadyCloned)) as any + } + if (typeof target === 'object' && target !== {}) { + const cp = { ...(target as { [key: string]: any }) } as { + [key: string]: any + } + Object.keys(cp).forEach(k => { + cp[k] = deepClone(cp[k], alreadyCloned) + }) + return cp as T + } + return target +} diff --git a/packages/botonic-plugin-contentful/tests/cms/cms.test.ts b/packages/botonic-plugin-contentful/tests/cms/cms.test.ts index cfa5ef7f33..4ff6965c27 100644 --- a/packages/botonic-plugin-contentful/tests/cms/cms.test.ts +++ b/packages/botonic-plugin-contentful/tests/cms/cms.test.ts @@ -1,6 +1,6 @@ -import { TOPCONTENT_TYPES, ContentType } from '../../src/cms' +import { TOP_CONTENT_TYPES, ContentType } from '../../src/cms' test('TEST: ALL_TYPES', () => { - expect(TOPCONTENT_TYPES.length).toBeGreaterThanOrEqual(9) - expect(TOPCONTENT_TYPES).toContain(ContentType.TEXT) + expect(TOP_CONTENT_TYPES.length).toBeGreaterThanOrEqual(8) + expect(TOP_CONTENT_TYPES).toContain(ContentType.TEXT) }) diff --git a/packages/botonic-plugin-contentful/tests/contentful/contentful.helper.ts b/packages/botonic-plugin-contentful/tests/contentful/contentful.helper.ts index 353c77ea23..0f508d8dd9 100644 --- a/packages/botonic-plugin-contentful/tests/contentful/contentful.helper.ts +++ b/packages/botonic-plugin-contentful/tests/contentful/contentful.helper.ts @@ -1,17 +1,29 @@ import Contentful from '../../src/contentful' import { Context } from '../../src/cms' +import { ContentfulOptions } from '../../src' export function testSpaceId(): string { return process.env.CONTENTFUL_TEST_SPACE_ID! } -export function testContentful(): Contentful { +export function testContentful( + options: Partial = {} +): Contentful { + return new Contentful(testContentfulOptions(options)) +} + +export function testContentfulOptions( + options: Partial = {} +): ContentfulOptions { // useful to have long timeouts so that we can send many requests simultaneously - return new Contentful({ - spaceId: testSpaceId(), - accessToken: process.env.CONTENTFUL_TEST_TOKEN!, - environment: 'master', - }) + return { + ...{ + spaceId: testSpaceId(), + accessToken: process.env.CONTENTFUL_TEST_TOKEN!, + environment: 'master', + }, + ...options, + } } export function testContext( diff --git a/packages/botonic-plugin-contentful/tests/contentful/contents.test.ts b/packages/botonic-plugin-contentful/tests/contentful/contents.test.ts index 1d9cee4235..16c75d45b5 100644 --- a/packages/botonic-plugin-contentful/tests/contentful/contents.test.ts +++ b/packages/botonic-plugin-contentful/tests/contentful/contents.test.ts @@ -10,6 +10,8 @@ import { } from '../../src/cms' import { testContentful, testContext } from './contentful.helper' +const BUTTON_POST_FAQ3 = '40buQOqp9jbwoxmMZhFO16' + test('TEST: contentful contents buttons', async () => { const buttons = await testContentful().contents(ContentType.BUTTON, { locale: 'en', @@ -26,10 +28,10 @@ test('TEST: contentful contents buttons', async () => { Callback.ofPayload('RATING_1') ) ) - const empezar = buttons.filter(b => b.id == '40buQOqp9jbwoxmMZhFO16') + const empezar = buttons.filter(b => b.id == BUTTON_POST_FAQ3) expect(empezar[0]).toEqual( new Button( - '40buQOqp9jbwoxmMZhFO16', + BUTTON_POST_FAQ3, 'POST_FAQ3', 'Return an article', new ContentCallback(ContentType.TEXT, 'C39lEROUgJl9hHSXKOEXS') diff --git a/packages/botonic-plugin-contentful/tests/contentful/contents/text.test.ts b/packages/botonic-plugin-contentful/tests/contentful/contents/text.test.ts index 199078ee5b..2f05e9c059 100644 --- a/packages/botonic-plugin-contentful/tests/contentful/contents/text.test.ts +++ b/packages/botonic-plugin-contentful/tests/contentful/contents/text.test.ts @@ -3,7 +3,7 @@ import { testContentful, testContext } from '../contentful.helper' import * as cms from '../../../src' export const TEST_POST_FAQ1_ID = 'djwHOFKknJ3AmyG6YKNip' -const TEST_POST_FAQ2_ID = '22h2Vba7v92MadcL5HeMrt' +export const TEST_POST_FAQ2_ID = '22h2Vba7v92MadcL5HeMrt' const TEST_FBK_MSG = '1U7XKJccDSsI3mP0yX04Mj' const TEST_FBK_OK_MSG = '63lakRZRu1AJ1DqlbZZb9O' const TEST_SORRY = '6ZjjdrKQbaLNc6JAhRnS8D' diff --git a/packages/botonic-plugin-contentful/tests/contentful/delivery.test.ts b/packages/botonic-plugin-contentful/tests/contentful/delivery.test.ts index 286eeb6207..997ba06149 100644 --- a/packages/botonic-plugin-contentful/tests/contentful/delivery.test.ts +++ b/packages/botonic-plugin-contentful/tests/contentful/delivery.test.ts @@ -1,4 +1,8 @@ -import { ContentCallback, ContentType, MESSAGE_TYPES } from '../../src/cms' +import { + ContentCallback, + ContentType, + MESSAGE_CONTENT_TYPES, +} from '../../src/cms' import { testContentful } from './contentful.helper' const TEST_IMAGE = '3xjvpC7d7PYBmiptEeygfd' @@ -6,7 +10,7 @@ const TEST_IMAGE = '3xjvpC7d7PYBmiptEeygfd' test('TEST: contentful delivery checks that we get the requested message type', async () => { const sut = testContentful() - for (const model of MESSAGE_TYPES) { + for (const model of MESSAGE_CONTENT_TYPES) { const callback = new ContentCallback(model, TEST_IMAGE) const content = callback.deliverPayloadContent(sut, { locale: 'es' }) if (model == ContentType.IMAGE) { diff --git a/packages/botonic-plugin-contentful/tests/contentful/ignore-fallback-decorator.test.ts b/packages/botonic-plugin-contentful/tests/contentful/ignore-fallback-decorator.test.ts new file mode 100644 index 0000000000..25bbb4cfa8 --- /dev/null +++ b/packages/botonic-plugin-contentful/tests/contentful/ignore-fallback-decorator.test.ts @@ -0,0 +1,121 @@ +import { testContentful, testContentfulOptions } from './contentful.helper' +import { AdaptorDeliveryApi } from '../../src/contentful/delivery-api' +import { createContentfulClientApi } from '../../src/contentful' +import { ButtonFields } from '../../src/contentful/contents/button' +import { ENGLISH, Locale, SPANISH } from '../../src/nlp' +import { + BOTONIC_CONTENT_TYPES, + ContentCallback, + ContentType, + Context, +} from '../../src/cms' +import { IgnoreFallbackDecorator } from '../../src/contentful/ignore-fallback-decorator' +import { TEST_POST_FAQ1_ID } from './contents/text.test' +import { TextFields } from '../../src/contentful/contents/text' +import { + TEST_CAROUSEL_MAIN_ID, + TEST_POST_MENU_CRSL, +} from './contents/carousel.test' + +const TEST_BUTTON_BLANK_SPANISH = '40buQOqp9jbwoxmMZhFO16' + +function createIgnoreFallbackDecorator() { + return new IgnoreFallbackDecorator( + new AdaptorDeliveryApi(createContentfulClientApi(testContentfulOptions())) + ) +} + +const FALLBACK_TESTS = [ + [{ locale: ENGLISH, ignoreFallbackLocale: true }, 'Return an article'], + [{ locale: SPANISH, ignoreFallbackLocale: true }, ''], + [{}, 'Return an article'], + [{ locale: ENGLISH }, 'Return an article'], + [{ locale: SPANISH }, 'Return an article'], // fallback +] + +test.each(FALLBACK_TESTS)( + 'TEST: IgnoreFallbackDecorator.getEntry uses fallback locale', + async (context: Context, expectedText: string) => { + const sut = createIgnoreFallbackDecorator() + + // act + const entry = await sut.getEntry( + TEST_BUTTON_BLANK_SPANISH, + context + ) + + // assert + expect(entry.fields.text).toEqual(expectedText) + expect(entry.fields.name).toEqual('POST_FAQ3') + } +) + +test('TEST: ignoreFallbackLocale with a carousel with buttons to other carousels ', async () => { + const contentful = testContentful() + const carousel = await contentful.carousel(TEST_CAROUSEL_MAIN_ID, { + locale: SPANISH, + ignoreFallbackLocale: true, + }) + const elementPost = carousel.elements.find( + e => e.id == '6aOPY3UVh8M4lihW9xe17E' + ) + expect(elementPost).toBeDefined() + expect(elementPost!.buttons).toHaveLength(1) + expect(elementPost!.buttons[0].callback).toEqual( + new ContentCallback(ContentType.CAROUSEL, TEST_POST_MENU_CRSL) + ) +}) + +test.each([ENGLISH, SPANISH])( + 'TEST: ignoreFallbackLocale all contents %s', + async (locale: Locale) => { + const contentful = testContentful() + for (const model of BOTONIC_CONTENT_TYPES) { + const ret = await contentful.contents(model, { + locale, + ignoreFallbackLocale: true, + }) + for (const content of ret) { + expect(content.id).toBeDefined() + expect(content.name).toBeDefined() + } + } + } +) + +test('TEST: IgnoreFallbackDecorator.getEntry uses fallback locale ', async () => { + const sut = createIgnoreFallbackDecorator() + + // act + const entry = await sut.getEntry(TEST_POST_FAQ1_ID, { + locale: SPANISH, + ignoreFallbackLocale: true, + }) + + // assert + expect(entry.fields.text).toEqual('') + expect(entry.fields.name).toEqual('POST_FAQ1') + expect(entry.fields.buttons.length).toEqual(1) +}) + +test.each(FALLBACK_TESTS)( + 'TEST: IgnoreFallbackDecorator.getEntries uses fallback locale', + async (context: Context, expectedText: string) => { + const sut = createIgnoreFallbackDecorator() + + // acr + const entries = await sut.getEntries(context, { + content_type: ContentType.BUTTON, + }) + + //assert + const entry = entries.items.filter( + e => e.sys.id == TEST_BUTTON_BLANK_SPANISH + ) + expect(entry.length).toBe(1) + expect(entry[0].fields.text).toEqual(expectedText) + expect(entry[0].fields.name).toEqual('POST_FAQ3') + } +) + +test('hack because webstorm does not recognize test.skip.each', () => {}) diff --git a/packages/botonic-plugin-contentful/tests/tools/translators/csv-export.test.ts b/packages/botonic-plugin-contentful/tests/tools/translators/csv-export.test.ts new file mode 100644 index 0000000000..920daedeb3 --- /dev/null +++ b/packages/botonic-plugin-contentful/tests/tools/translators/csv-export.test.ts @@ -0,0 +1,76 @@ +import { + ContentToCsvLines, + CsvExport, + skipEmptyStrings, +} from '../../../src/tools/translators/csv-export' +import { testContentful } from '../../../tests/contentful/contentful.helper' +import { ENGLISH } from '../../../src/nlp' +import { TextBuilder } from '../../../src/cms/factories' +import sync from 'csv-stringify/lib/sync' +import { RndButtonsBuilder } from '../../../src/cms/test-helpers' + +test('TEST: CsvExport integration test', async () => { + const cms = testContentful() + + const exporter = new CsvExport({}) + // running for ENGLISH to test contents with empty fields + const from = ENGLISH + await exporter.write(`/tmp/contentful_${from}.csv`, cms, from) +}) + +test('TEST: ContentToCsvLines.getCsvLines Text', () => { + const exporter = new ContentToCsvLines({}) + const fields = exporter.getCsvLines( + new TextBuilder('id1', 'name1', 'long text').withShortText('short1').build() + ) + expect(fields).toEqual([ + ['text', 'name1', 'id1', 'Short text', 'short1'], + ['text', 'name1', 'id1', 'Text', 'long text'], + ]) +}) + +test('TEST: ContentToCsvLines.getI18nFields skips blank and undefined strings', () => { + const exporter = new ContentToCsvLines({ stringFilter: skipEmptyStrings }) + const textWithoutShortText = new TextBuilder('id1', 'name1', ' ').build() + const fields = exporter.getCsvLines(textWithoutShortText) + expect(fields).toEqual([]) +}) + +test('TEST: ContentToCsvLines.getI18nFields Button', () => { + const exporter = new ContentToCsvLines({}) + const button = new RndButtonsBuilder().addButton().build()[0] + const fields = exporter.getCsvLines(button) + expect(fields).toEqual([ + ['button', button.name, button.id, 'Text', button.text], + ]) +}) + +// Skipped because push callback within sync is never called (why?), and hence it returns "" +test.skip('TEST: CsvExport.stringifier sync', () => { + const exporter = new CsvExport({ + nameFilter: n => ['HOME_RETURN_URL'].includes(n), + }) + const stringifier = exporter.create_stringifier() + const str = sync(['1 "2" 3', '4'], stringifier.options) + expect(str).toEqual('"1 ""2"" 3","4"') +}) + +test('TEST: CsvExport.stringifier', () => { + const exporter = new CsvExport({ + nameFilter: n => ['HOME_RETURN_URL'].includes(n), + }) + const fields = [] + // does not work with Readable.from + // const readable = Readable.from(['1 "2" 3', '4']).pipe(exporter.stringifier) + const stringifier = exporter.create_stringifier() + stringifier.write(['1 "2" 3', '4']) + stringifier.end() + let row: Buffer + while ((row = stringifier.read())) { + fields.push(row.toString('utf8')) + } + + expect(fields).toEqual([ + '"Model";"Code";"Id";"Field";"From";"To"\r\n"1 ""2"" 3";"4"\r\n', + ]) +}) diff --git a/packages/botonic-plugin-contentful/tests/util/objects.test.ts b/packages/botonic-plugin-contentful/tests/util/objects.test.ts new file mode 100644 index 0000000000..1853eb9214 --- /dev/null +++ b/packages/botonic-plugin-contentful/tests/util/objects.test.ts @@ -0,0 +1,55 @@ +import { deepClone, shallowClone } from '../../src/util/objects' + +class Subclass { + constructor(public field: number) {} +} +class Class { + constructor(public arr = [3, 4], public sub = new Subclass(1)) {} +} + +test('TEST: shallowClone', () => { + const source = new Class() + + // act + const copy = shallowClone(source) + + // assert + expect(copy).toEqual(source) + expect(copy).not.toBe(source) + + source.sub = new Subclass(2) + expect(copy).not.toEqual(source) + + copy.sub.field = 3 + expect(source.sub.field).toEqual(2) +}) + +test('TEST: deepClone', () => { + const source = new Class() + + // act + const copy = deepClone(source) + + // assert + expect(copy).toEqual(source) + expect(copy).not.toBe(source) + + source.sub = new Subclass(2) + expect(copy).not.toEqual(source) + + copy.sub.field = 3 + expect(source.sub.field).not.toEqual(3) +}) + +class RecursiveClass { + constructor(public rec?: RecursiveClass) {} +} + +test('TEST: deepClone no recursive call', () => { + const source = new RecursiveClass() + source.rec = source + + // act + const copy = deepClone(source) + expect(copy).toEqual(source) +}) diff --git a/packages/botonic-react/src/components/multichannel/index.d.ts b/packages/botonic-react/src/components/multichannel/index.d.ts index c492e0a15c..003e2e7ee6 100644 --- a/packages/botonic-react/src/components/multichannel/index.d.ts +++ b/packages/botonic-react/src/components/multichannel/index.d.ts @@ -1,8 +1,8 @@ import * as React from 'react' export const Multichannel: React.FunctionComponent<{ - firstIndex: number - boldIndex: boolean + firstIndex?: number + boldIndex?: boolean }> // Text