Skip to content

Commit

Permalink
fix: type compatible with v9 tag definition, support Context as sco…
Browse files Browse the repository at this point in the history
…pe in various render APIs, #570
  • Loading branch information
harttle committed Dec 18, 2022
1 parent 7526d40 commit fb6a9f8
Show file tree
Hide file tree
Showing 5 changed files with 55 additions and 20 deletions.
6 changes: 5 additions & 1 deletion src/liquid-options.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { assert, isArray, isString, isFunction } from './util'
import { LRU, LiquidCache } from './cache'
import { FS } from './fs/fs'
import { FS, LookupType } from './fs'
import * as fs from './fs/node'
import { defaultOperators, Operators } from './render'
import { json } from './filters/misc'
Expand Down Expand Up @@ -91,6 +91,10 @@ export interface RenderOptions {
ownPropertyOnly?: boolean;
}

export interface RenderFileOptions extends RenderOptions {
lookupType?: LookupType;
}

interface NormalizedOptions extends LiquidOptions {
root?: string[];
partials?: string[];
Expand Down
37 changes: 21 additions & 16 deletions src/liquid.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { Render } from './render'
import { Parser } from './parser'
import { tags } from './tags'
import { filters } from './filters'
import { LiquidOptions, normalizeDirectoryList, NormalizedFullOptions, normalize, RenderOptions } from './liquid-options'
import { LiquidOptions, normalizeDirectoryList, NormalizedFullOptions, normalize, RenderOptions, RenderFileOptions } from './liquid-options'

export class Liquid {
public readonly options: NormalizedFullOptions
Expand All @@ -25,8 +25,8 @@ export class Liquid {
return this.parser.parse(html, filepath)
}

public _render (tpl: Template[], scope: object | undefined, renderOptions: RenderOptions): IterableIterator<any> {
const ctx = new Context(scope, this.options, renderOptions)
public _render (tpl: Template[], scope: Context | object | undefined, renderOptions: RenderOptions): IterableIterator<any> {
const ctx = scope instanceof Context ? scope : new Context(scope, this.options, renderOptions)
return this.renderer.renderTemplates(tpl, ctx)
}
public async render (tpl: Template[], scope?: object, renderOptions?: RenderOptions): Promise<any> {
Expand All @@ -40,14 +40,14 @@ export class Liquid {
return this.renderer.renderTemplatesToNodeStream(tpl, ctx)
}

public _parseAndRender (html: string, scope: object | undefined, renderOptions: RenderOptions): IterableIterator<any> {
public _parseAndRender (html: string, scope: Context | object | undefined, renderOptions: RenderOptions): IterableIterator<any> {
const tpl = this.parse(html)
return this._render(tpl, scope, renderOptions)
}
public async parseAndRender (html: string, scope?: object, renderOptions?: RenderOptions): Promise<any> {
public async parseAndRender (html: string, scope?: Context | object, renderOptions?: RenderOptions): Promise<any> {
return toPromise(this._parseAndRender(html, scope, { ...renderOptions, sync: false }))
}
public parseAndRenderSync (html: string, scope?: object, renderOptions?: RenderOptions): any {
public parseAndRenderSync (html: string, scope?: Context | object, renderOptions?: RenderOptions): any {
return toValueSync(this._parseAndRender(html, scope, { ...renderOptions, sync: true }))
}

Expand All @@ -57,19 +57,24 @@ export class Liquid {
public _parseLayoutFile (file: string, sync?: boolean, currentFile?: string) {
return this.parser.parseFile(file, sync, LookupType.Layouts, currentFile)
}
public async parseFile (file: string): Promise<Template[]> {
return toPromise<Template[]>(this.parser.parseFile(file, false))
public _parseFile (file: string, sync?: boolean, lookupType?: LookupType, currentFile?: string): Generator<unknown, Template[]> {
return this.parser.parseFile(file, sync, lookupType, currentFile)
}
public parseFileSync (file: string): Template[] {
return toValueSync<Template[]>(this.parser.parseFile(file, true))
public async parseFile (file: string, lookupType?: LookupType): Promise<Template[]> {
return toPromise<Template[]>(this.parser.parseFile(file, false, lookupType))
}
public async renderFile (file: string, ctx?: object, renderOptions?: RenderOptions) {
const templates = await this.parseFile(file)
return this.render(templates, ctx, renderOptions)
public parseFileSync (file: string, lookupType?: LookupType): Template[] {
return toValueSync<Template[]>(this.parser.parseFile(file, true, lookupType))
}
public * _renderFile (file: string, ctx: Context | object | undefined, renderFileOptions: RenderFileOptions): Generator<any> {
const templates = (yield this._parseFile(file, renderFileOptions.sync, renderFileOptions.lookupType)) as Template[]
return yield this._render(templates, ctx, renderFileOptions)
}
public async renderFile (file: string, ctx?: Context | object, renderFileOptions?: RenderFileOptions) {
return toPromise(this._renderFile(file, ctx, { ...renderFileOptions, sync: false }))
}
public renderFileSync (file: string, ctx?: object, renderOptions?: RenderOptions) {
const templates = this.parseFileSync(file)
return this.renderSync(templates, ctx, renderOptions)
public renderFileSync (file: string, ctx?: Context | object, renderFileOptions?: RenderFileOptions) {
return toValueSync(this._renderFile(file, ctx, { ...renderFileOptions, sync: true }))
}
public async renderFileToNodeStream (file: string, scope?: object, renderOptions?: RenderOptions) {
const templates = await this.parseFile(file)
Expand Down
5 changes: 3 additions & 2 deletions src/template/tag-options-adapter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,9 @@ import { Context } from '../context'
import type { Liquid } from '../liquid'

export interface TagImplOptions {
parse?: (this: Tag, token: TagToken, remainingTokens: TopLevelToken[]) => void;
render: (this: Tag, ctx: Context, emitter: Emitter, hash: Record<string, any>) => TagRenderReturn;
[key: string]: any
parse?: (this: Tag & TagImplOptions, token: TagToken, remainingTokens: TopLevelToken[]) => void;
render: (this: Tag & TagImplOptions, ctx: Context, emitter: Emitter, hash: Record<string, any>) => TagRenderReturn;
}

export function createTagClass (options: TagImplOptions): TagClass {
Expand Down
18 changes: 17 additions & 1 deletion test/e2e/issues.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { Tokenizer, Context, Liquid, Drop, toValueSync } from '../..'
import { TopLevelToken, TagToken, Tokenizer, Context, Liquid, Drop, toValueSync } from '../..'
import { expect, use } from 'chai'
import * as chaiAsPromised from 'chai-as-promised'
import * as sinon from 'sinon'
Expand Down Expand Up @@ -346,4 +346,20 @@ describe('Issues', function () {
const html = await liquid.parseAndRender(tpl)
expect(html).to.match(/^\s*This is a love or luck potion.\s+This is a strength or health or love potion.\s*$/)
})
it('#570 tag registration compatible to v9', async () => {
const liquid = new Liquid()
liquid.registerTag('metadata_file', {
parse (tagToken: TagToken, remainTokens: TopLevelToken[]) {
this.str = tagToken.args
},
async render (ctx: Context) {
const content = await Promise.resolve(`{{${this.str}}}`)
return this.liquid.parseAndRender(content.toString(), ctx)
}
})
const tpl = '{% metadata_file foo %}'
const ctx = { foo: 'FOO' }
const html = await liquid.parseAndRender(tpl, ctx)
expect(html).to.equal('FOO')
})
})
9 changes: 9 additions & 0 deletions test/integration/liquid/liquid.ts
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,10 @@ describe('Liquid', function () {
const html = await engine.parseAndRender(src, { foo: Promise.resolve('FOO') })
expect(html).to.equal('FOO')
})
it('should parse and render with Context', async function () {
const html = await engine.parseAndRender('{{foo}}', new Context({ foo: 'FOO' }))
expect(html).to.equal('FOO')
})
})
describe('#parseAndRenderSync', function () {
const engine = new Liquid()
Expand Down Expand Up @@ -145,6 +149,11 @@ describe('Liquid', function () {
const str = await engine.evalValue('"foo"', ctx)
expect(str).to.equal('foo')
})
it('should support plain scope', async function () {
const engine = new Liquid()
const str = await engine.evalValue('foo', { foo: 'FOO' })
expect(str).to.equal('FOO')
})
})
describe('#evalValueSync', function () {
it('should eval string literal', function () {
Expand Down

0 comments on commit fb6a9f8

Please sign in to comment.