@app-config/core
Advanced tools
Comparing version 2.5.0 to 2.5.1
import { Json } from '@app-config/utils'; | ||
import { ParsedValue, ParsingExtension } from './parsed-value'; | ||
import { ParsedValue, ParsingContext, ParsingExtension } from './parsed-value'; | ||
export declare enum FileType { | ||
@@ -18,5 +18,5 @@ YAML = "YAML", | ||
/** Reads the contents of the source into a full ParsedValue (not the raw JSON, like readValue) */ | ||
read(extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
read(extensions?: ParsingExtension[], context?: ParsingContext): Promise<ParsedValue>; | ||
/** Ergonomic helper for chaining `source.read(extensions).then(v => v.toJSON())` */ | ||
readToJSON(extensions?: ParsingExtension[]): Promise<Json>; | ||
readToJSON(extensions?: ParsingExtension[], context?: ParsingContext): Promise<Json>; | ||
} | ||
@@ -36,3 +36,3 @@ /** Read configuration from a literal JS object */ | ||
readValue(): Promise<Json>; | ||
read(extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
read(extensions?: ParsingExtension[], context?: ParsingContext): Promise<ParsedValue>; | ||
} | ||
@@ -45,3 +45,3 @@ /** Read configuration from the first ConfigSource that doesn't fail */ | ||
readValue(): Promise<Json>; | ||
read(extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
read(extensions?: ParsingExtension[], context?: ParsingContext): Promise<ParsedValue>; | ||
} | ||
@@ -48,0 +48,0 @@ export declare function stringify(config: Json, fileType: FileType, minimal?: boolean): string; |
@@ -28,9 +28,9 @@ "use strict"; | ||
/** Reads the contents of the source into a full ParsedValue (not the raw JSON, like readValue) */ | ||
async read(extensions) { | ||
async read(extensions, context) { | ||
const rawValue = await this.readValue(); | ||
return parsed_value_1.ParsedValue.parse(rawValue, this, extensions); | ||
return parsed_value_1.ParsedValue.parse(rawValue, this, extensions, undefined, context); | ||
} | ||
/** Ergonomic helper for chaining `source.read(extensions).then(v => v.toJSON())` */ | ||
async readToJSON(extensions) { | ||
const parsed = await this.read(extensions); | ||
async readToJSON(extensions, context) { | ||
const parsed = await this.read(extensions, context); | ||
return parsed.toJSON(); | ||
@@ -73,4 +73,4 @@ } | ||
// override so that ParsedValue is directly from the originating ConfigSource | ||
async read(extensions) { | ||
const values = await Promise.all(this.sources.map((source) => source.read(extensions))); | ||
async read(extensions, context) { | ||
const values = await Promise.all(this.sources.map((source) => source.read(extensions, Object.assign({}, context)))); | ||
const merged = values.reduce((acc, parsed) => { | ||
@@ -121,7 +121,7 @@ if (!acc) | ||
// override so that ParsedValue is directly from the originating ConfigSource | ||
async read(extensions) { | ||
async read(extensions, context) { | ||
// take the first value that comes back without an error | ||
for (const source of this.sources) { | ||
try { | ||
const value = await source.read(extensions); | ||
const value = await source.read(extensions, context); | ||
logging_1.logger.verbose(`FallbackSource found successful source`); | ||
@@ -128,0 +128,0 @@ return value; |
import { Json } from '@app-config/utils'; | ||
import { ParsedValue, ParsingExtension } from './parsed-value'; | ||
import { ParsedValue, ParsingContext, ParsingExtension } from './parsed-value'; | ||
export declare enum FileType { | ||
@@ -18,5 +18,5 @@ YAML = "YAML", | ||
/** Reads the contents of the source into a full ParsedValue (not the raw JSON, like readValue) */ | ||
read(extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
read(extensions?: ParsingExtension[], context?: ParsingContext): Promise<ParsedValue>; | ||
/** Ergonomic helper for chaining `source.read(extensions).then(v => v.toJSON())` */ | ||
readToJSON(extensions?: ParsingExtension[]): Promise<Json>; | ||
readToJSON(extensions?: ParsingExtension[], context?: ParsingContext): Promise<Json>; | ||
} | ||
@@ -36,3 +36,3 @@ /** Read configuration from a literal JS object */ | ||
readValue(): Promise<Json>; | ||
read(extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
read(extensions?: ParsingExtension[], context?: ParsingContext): Promise<ParsedValue>; | ||
} | ||
@@ -45,3 +45,3 @@ /** Read configuration from the first ConfigSource that doesn't fail */ | ||
readValue(): Promise<Json>; | ||
read(extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
read(extensions?: ParsingExtension[], context?: ParsingContext): Promise<ParsedValue>; | ||
} | ||
@@ -48,0 +48,0 @@ export declare function stringify(config: Json, fileType: FileType, minimal?: boolean): string; |
@@ -25,9 +25,9 @@ import { extname } from 'path'; | ||
/** Reads the contents of the source into a full ParsedValue (not the raw JSON, like readValue) */ | ||
async read(extensions) { | ||
async read(extensions, context) { | ||
const rawValue = await this.readValue(); | ||
return ParsedValue.parse(rawValue, this, extensions); | ||
return ParsedValue.parse(rawValue, this, extensions, undefined, context); | ||
} | ||
/** Ergonomic helper for chaining `source.read(extensions).then(v => v.toJSON())` */ | ||
async readToJSON(extensions) { | ||
const parsed = await this.read(extensions); | ||
async readToJSON(extensions, context) { | ||
const parsed = await this.read(extensions, context); | ||
return parsed.toJSON(); | ||
@@ -68,4 +68,4 @@ } | ||
// override so that ParsedValue is directly from the originating ConfigSource | ||
async read(extensions) { | ||
const values = await Promise.all(this.sources.map((source) => source.read(extensions))); | ||
async read(extensions, context) { | ||
const values = await Promise.all(this.sources.map((source) => source.read(extensions, { ...context }))); | ||
const merged = values.reduce((acc, parsed) => { | ||
@@ -115,7 +115,7 @@ if (!acc) | ||
// override so that ParsedValue is directly from the originating ConfigSource | ||
async read(extensions) { | ||
async read(extensions, context) { | ||
// take the first value that comes back without an error | ||
for (const source of this.sources) { | ||
try { | ||
const value = await source.read(extensions); | ||
const value = await source.read(extensions, context); | ||
logger.verbose(`FallbackSource found successful source`); | ||
@@ -122,0 +122,0 @@ return value; |
@@ -12,4 +12,7 @@ /// <reference types="node" /> | ||
export declare type ParsingExtensionKey = [typeof InObject, string] | [typeof InArray, number] | [typeof Root]; | ||
export interface ParsingContext { | ||
[k: string]: string | string[] | undefined | ParsingContext; | ||
} | ||
export interface ParsingExtension { | ||
(value: Json, key: ParsingExtensionKey, context: ParsingExtensionKey[]): ParsingExtensionTransform | false; | ||
(value: Json, key: ParsingExtensionKey, parentKeys: ParsingExtensionKey[], context: ParsingContext): ParsingExtensionTransform | false; | ||
/** | ||
@@ -22,3 +25,3 @@ * A globally unique string that identifies what parsing extension this is. | ||
} | ||
export declare type ParsingExtensionTransform = (parse: (value: Json, metadata?: ParsedValueMetadata, source?: ConfigSource, extensions?: ParsingExtension[]) => Promise<ParsedValue>, parent: JsonObject | Json[] | undefined, source: ConfigSource, extensions: ParsingExtension[], root: Json) => Promise<ParsedValue> | ParsedValue; | ||
export declare type ParsingExtensionTransform = (parse: (value: Json, metadata?: ParsedValueMetadata, source?: ConfigSource, extensions?: ParsingExtension[], context?: ParsingContext) => Promise<ParsedValue>, parent: JsonObject | Json[] | undefined, source: ConfigSource, extensions: ParsingExtension[], root: Json) => Promise<ParsedValue> | ParsedValue; | ||
export interface ParsedValueMetadata { | ||
@@ -40,3 +43,3 @@ [key: string]: any; | ||
/** Parses (with extensions) from a value that was read from ConfigSource */ | ||
static parse(raw: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata): Promise<ParsedValue>; | ||
static parse(raw: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata, context?: ParsingContext): Promise<ParsedValue>; | ||
/** Parses (with extensions) from a plain JSON object */ | ||
@@ -67,3 +70,3 @@ static parseLiteral(raw: Json, extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
} | ||
export declare function parseValue(value: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata): Promise<ParsedValue>; | ||
export declare function parseValue(value: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata, context?: ParsingContext): Promise<ParsedValue>; | ||
export {}; |
@@ -26,4 +26,4 @@ import { inspect } from 'util'; | ||
/** Parses (with extensions) from a value that was read from ConfigSource */ | ||
static async parse(raw, source, extensions, metadata) { | ||
return parseValue(raw, source, extensions, metadata); | ||
static async parse(raw, source, extensions, metadata, context) { | ||
return parseValue(raw, source, extensions, metadata, context); | ||
} | ||
@@ -239,8 +239,8 @@ /** Parses (with extensions) from a plain JSON object */ | ||
} | ||
export async function parseValue(value, source, extensions = [], metadata = {}) { | ||
return parseValueInner(value, source, extensions, metadata, [[Root]], value); | ||
export async function parseValue(value, source, extensions = [], metadata = {}, context = {}) { | ||
return parseValueInner(value, source, extensions, metadata, context, [[Root]], value); | ||
} | ||
async function parseValueInner(value, source, extensions, metadata = {}, context, root, parent, visitedExtensions = new Set()) { | ||
const [currentKey] = context.slice(-1); | ||
const contextualKeys = context.slice(0, context.length - 1); | ||
async function parseValueInner(value, source, extensions, metadata = {}, context = {}, parentKeys, root, parent, visitedExtensions = new Set()) { | ||
const [currentKey] = parentKeys.slice(-1); | ||
const parentKeysNext = parentKeys.slice(0, parentKeys.length - 1); | ||
let applicableExtension; | ||
@@ -260,3 +260,3 @@ // before anything else, we check for parsing extensions that should be applied | ||
continue; | ||
const applicable = extension(value, currentKey, contextualKeys); | ||
const applicable = extension(value, currentKey, parentKeysNext, context); | ||
if (applicable) { | ||
@@ -272,3 +272,3 @@ applicableExtension = applicable; | ||
if (applicableExtension) { | ||
const parse = (inner, metadataOverride, sourceOverride, extensionsOverride) => parseValueInner(inner, sourceOverride ?? source, extensionsOverride ?? extensions, { ...metadata, ...metadataOverride }, context, root, parent, visitedExtensions); | ||
const parse = (inner, metadataOverride, sourceOverride, extensionsOverride, contextOverride) => parseValueInner(inner, sourceOverride ?? source, extensionsOverride ?? extensions, { ...metadata, ...metadataOverride }, { ...context, ...contextOverride }, parentKeys, root, parent, visitedExtensions); | ||
// note that we don't traverse the object is an extension applied, that's up to them (with `parse`) | ||
@@ -279,3 +279,3 @@ return applicableExtension(parse, parent, source, extensions, root); | ||
const output = await Promise.all(Array.from(value.entries()).map(([index, item]) => { | ||
return parseValueInner(item, source, extensions, undefined, context.concat([[InArray, index]]), root, value); | ||
return parseValueInner(item, source, extensions, undefined, context, parentKeys.concat([[InArray, index]]), root, value); | ||
})); | ||
@@ -291,3 +291,3 @@ return new ParsedValue(source, value, output).assignMeta(metadata); | ||
await Promise.all(Object.entries(value).map(async ([key, item]) => { | ||
const parsed = await parseValueInner(item, source, extensions, undefined, context.concat([[InObject, key]]), root, value); | ||
const parsed = await parseValueInner(item, source, extensions, undefined, context, parentKeys.concat([[InObject, key]]), root, value); | ||
// NOTE: shouldMerge is treated as shouldFlatten when the value itself is not an object (because we cannot merge arrays or primitives) | ||
@@ -294,0 +294,0 @@ if (parsed.meta.shouldFlatten) { |
@@ -12,4 +12,7 @@ /// <reference types="node" /> | ||
export declare type ParsingExtensionKey = [typeof InObject, string] | [typeof InArray, number] | [typeof Root]; | ||
export interface ParsingContext { | ||
[k: string]: string | string[] | undefined | ParsingContext; | ||
} | ||
export interface ParsingExtension { | ||
(value: Json, key: ParsingExtensionKey, context: ParsingExtensionKey[]): ParsingExtensionTransform | false; | ||
(value: Json, key: ParsingExtensionKey, parentKeys: ParsingExtensionKey[], context: ParsingContext): ParsingExtensionTransform | false; | ||
/** | ||
@@ -22,3 +25,3 @@ * A globally unique string that identifies what parsing extension this is. | ||
} | ||
export declare type ParsingExtensionTransform = (parse: (value: Json, metadata?: ParsedValueMetadata, source?: ConfigSource, extensions?: ParsingExtension[]) => Promise<ParsedValue>, parent: JsonObject | Json[] | undefined, source: ConfigSource, extensions: ParsingExtension[], root: Json) => Promise<ParsedValue> | ParsedValue; | ||
export declare type ParsingExtensionTransform = (parse: (value: Json, metadata?: ParsedValueMetadata, source?: ConfigSource, extensions?: ParsingExtension[], context?: ParsingContext) => Promise<ParsedValue>, parent: JsonObject | Json[] | undefined, source: ConfigSource, extensions: ParsingExtension[], root: Json) => Promise<ParsedValue> | ParsedValue; | ||
export interface ParsedValueMetadata { | ||
@@ -40,3 +43,3 @@ [key: string]: any; | ||
/** Parses (with extensions) from a value that was read from ConfigSource */ | ||
static parse(raw: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata): Promise<ParsedValue>; | ||
static parse(raw: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata, context?: ParsingContext): Promise<ParsedValue>; | ||
/** Parses (with extensions) from a plain JSON object */ | ||
@@ -67,3 +70,3 @@ static parseLiteral(raw: Json, extensions?: ParsingExtension[]): Promise<ParsedValue>; | ||
} | ||
export declare function parseValue(value: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata): Promise<ParsedValue>; | ||
export declare function parseValue(value: Json, source: ConfigSource, extensions?: ParsingExtension[], metadata?: ParsedValueMetadata, context?: ParsingContext): Promise<ParsedValue>; | ||
export {}; |
@@ -32,4 +32,4 @@ "use strict"; | ||
/** Parses (with extensions) from a value that was read from ConfigSource */ | ||
static async parse(raw, source, extensions, metadata) { | ||
return parseValue(raw, source, extensions, metadata); | ||
static async parse(raw, source, extensions, metadata, context) { | ||
return parseValue(raw, source, extensions, metadata, context); | ||
} | ||
@@ -248,9 +248,9 @@ /** Parses (with extensions) from a plain JSON object */ | ||
} | ||
async function parseValue(value, source, extensions = [], metadata = {}) { | ||
return parseValueInner(value, source, extensions, metadata, [[exports.Root]], value); | ||
async function parseValue(value, source, extensions = [], metadata = {}, context = {}) { | ||
return parseValueInner(value, source, extensions, metadata, context, [[exports.Root]], value); | ||
} | ||
exports.parseValue = parseValue; | ||
async function parseValueInner(value, source, extensions, metadata = {}, context, root, parent, visitedExtensions = new Set()) { | ||
const [currentKey] = context.slice(-1); | ||
const contextualKeys = context.slice(0, context.length - 1); | ||
async function parseValueInner(value, source, extensions, metadata = {}, context = {}, parentKeys, root, parent, visitedExtensions = new Set()) { | ||
const [currentKey] = parentKeys.slice(-1); | ||
const parentKeysNext = parentKeys.slice(0, parentKeys.length - 1); | ||
let applicableExtension; | ||
@@ -270,3 +270,3 @@ // before anything else, we check for parsing extensions that should be applied | ||
continue; | ||
const applicable = extension(value, currentKey, contextualKeys); | ||
const applicable = extension(value, currentKey, parentKeysNext, context); | ||
if (applicable) { | ||
@@ -282,3 +282,3 @@ applicableExtension = applicable; | ||
if (applicableExtension) { | ||
const parse = (inner, metadataOverride, sourceOverride, extensionsOverride) => parseValueInner(inner, sourceOverride !== null && sourceOverride !== void 0 ? sourceOverride : source, extensionsOverride !== null && extensionsOverride !== void 0 ? extensionsOverride : extensions, Object.assign(Object.assign({}, metadata), metadataOverride), context, root, parent, visitedExtensions); | ||
const parse = (inner, metadataOverride, sourceOverride, extensionsOverride, contextOverride) => parseValueInner(inner, sourceOverride !== null && sourceOverride !== void 0 ? sourceOverride : source, extensionsOverride !== null && extensionsOverride !== void 0 ? extensionsOverride : extensions, Object.assign(Object.assign({}, metadata), metadataOverride), Object.assign(Object.assign({}, context), contextOverride), parentKeys, root, parent, visitedExtensions); | ||
// note that we don't traverse the object is an extension applied, that's up to them (with `parse`) | ||
@@ -289,3 +289,3 @@ return applicableExtension(parse, parent, source, extensions, root); | ||
const output = await Promise.all(Array.from(value.entries()).map(([index, item]) => { | ||
return parseValueInner(item, source, extensions, undefined, context.concat([[exports.InArray, index]]), root, value); | ||
return parseValueInner(item, source, extensions, undefined, context, parentKeys.concat([[exports.InArray, index]]), root, value); | ||
})); | ||
@@ -301,3 +301,3 @@ return new ParsedValue(source, value, output).assignMeta(metadata); | ||
await Promise.all(Object.entries(value).map(async ([key, item]) => { | ||
const parsed = await parseValueInner(item, source, extensions, undefined, context.concat([[exports.InObject, key]]), root, value); | ||
const parsed = await parseValueInner(item, source, extensions, undefined, context, parentKeys.concat([[exports.InObject, key]]), root, value); | ||
// NOTE: shouldMerge is treated as shouldFlatten when the value itself is not an object (because we cannot merge arrays or primitives) | ||
@@ -304,0 +304,0 @@ if (parsed.meta.shouldFlatten) { |
{ | ||
"name": "@app-config/core", | ||
"description": "Core logic for App Config", | ||
"version": "2.5.0", | ||
"version": "2.5.1", | ||
"license": "MPL-2.0", | ||
@@ -33,4 +33,4 @@ "author": { | ||
"dependencies": { | ||
"@app-config/logging": "^2.5.0", | ||
"@app-config/utils": "^2.5.0", | ||
"@app-config/logging": "^2.5.1", | ||
"@app-config/utils": "^2.5.1", | ||
"@iarna/toml": "3", | ||
@@ -42,3 +42,3 @@ "js-yaml": "^3.13.1", | ||
"devDependencies": { | ||
"@app-config/test-utils": "^2.5.0", | ||
"@app-config/test-utils": "^2.5.1", | ||
"@types/js-yaml": "3", | ||
@@ -45,0 +45,0 @@ "@types/lodash.merge": "4" |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
114750
1649
Updated@app-config/logging@^2.5.1
Updated@app-config/utils@^2.5.1