Last active
May 16, 2020 21:27
-
-
Save mikilian/d7b589ee50dc24672d1cba212f0ab542 to your computer and use it in GitHub Desktop.
reflect-metadata.ts 4.1.2 for deno
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/*! ***************************************************************************** | |
Copyright (C) Microsoft. All rights reserved. | |
Licensed under the Apache License, Version 2.0 (the "License"); you may not use | |
this file except in compliance with the License. You may obtain a copy of the | |
License at http://www.apache.org/licenses/LICENSE-2.0 | |
THIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | |
KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED | |
WARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE, | |
MERCHANTABLITY OR NON-INFRINGEMENT. | |
See the Apache Version 2.0 License for specific language governing permissions | |
and limitations under the License. | |
***************************************************************************** */ | |
export namespace Reflect { | |
// Metadata Proposal | |
// https://rbuckton.github.io/reflect-metadata/ | |
type HashMap<V> = Record<string, V>; | |
interface BufferLike { | |
[offset: number]: number; | |
length: number; | |
} | |
type IteratorResult<T> = { value: T, done: false } | { value: never, done: true }; | |
interface Iterator<T> { | |
next(value?: any): IteratorResult<T>; | |
throw?(value: any): IteratorResult<T>; | |
return?(value?: T): IteratorResult<T>; | |
} | |
interface Iterable<T> { | |
"@@iterator"(): Iterator<T>; | |
} | |
interface IterableIterator<T> extends Iterator<T> { | |
"@@iterator"(): IterableIterator<T>; | |
} | |
interface Map<K, V> extends Iterable<[K, V]> { | |
size: number; | |
has(key: K): boolean; | |
get(key: K): V; | |
set(key: K, value?: V): this; | |
delete(key: K): boolean; | |
clear(): void; | |
keys(): IterableIterator<K>; | |
values(): IterableIterator<V>; | |
entries(): IterableIterator<[K, V]>; | |
} | |
interface MapConstructor { | |
new (): Map<any, any>; | |
new <K, V>(): Map<K, V>; | |
prototype: Map<any, any>; | |
} | |
interface Set<T> extends Iterable<T> { | |
size: number; | |
has(value: T): boolean; | |
add(value: T): this; | |
delete(value: T): boolean; | |
clear(): void; | |
keys(): IterableIterator<T>; | |
values(): IterableIterator<T>; | |
entries(): IterableIterator<[T, T]>; | |
} | |
interface SetConstructor { | |
new (): Set<any>; | |
new <T>(): Set<T>; | |
prototype: Set<any>; | |
} | |
interface WeakMap<K, V> { | |
clear(): void; | |
delete(key: K): boolean; | |
get(key: K): V; | |
has(key: K): boolean; | |
set(key: K, value?: V): WeakMap<K, V>; | |
} | |
interface WeakMapConstructor { | |
new (): WeakMap<any, any>; | |
new <K, V>(): WeakMap<K, V>; | |
prototype: WeakMap<any, any>; | |
} | |
type MemberDecorator = <T>(target: Object, propertyKey: string | symbol, descriptor?: TypedPropertyDescriptor<T>) => TypedPropertyDescriptor<T> | void; | |
declare const Symbol: { iterator: symbol, toPrimitive: symbol }; | |
declare const Set: SetConstructor; | |
declare const WeakMap: WeakMapConstructor; | |
declare const Map: MapConstructor; | |
declare const global: any; | |
declare const crypto: Crypto; | |
declare const msCrypto: Crypto; | |
declare const process: any; | |
/** | |
* Applies a set of decorators to a target object. | |
* @param decorators An array of decorators. | |
* @param target The target object. | |
* @returns The result of applying the provided decorators. | |
* @remarks Decorators are applied in reverse order of their positions in the array. | |
* @example | |
* | |
* class Example { } | |
* | |
* // constructor | |
* Example = Reflect.decorate(decoratorsArray, Example); | |
* | |
*/ | |
export declare function decorate(decorators: ClassDecorator[], target: Function): Function; | |
/** | |
* Applies a set of decorators to a property of a target object. | |
* @param decorators An array of decorators. | |
* @param target The target object. | |
* @param propertyKey The property key to decorate. | |
* @param attributes A property descriptor. | |
* @remarks Decorators are applied in reverse order. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod() { } | |
* method() { } | |
* } | |
* | |
* // property (on constructor) | |
* Reflect.decorate(decoratorsArray, Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* Reflect.decorate(decoratorsArray, Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* Object.defineProperty(Example, "staticMethod", | |
* Reflect.decorate(decoratorsArray, Example, "staticMethod", | |
* Object.getOwnPropertyDescriptor(Example, "staticMethod"))); | |
* | |
* // method (on prototype) | |
* Object.defineProperty(Example.prototype, "method", | |
* Reflect.decorate(decoratorsArray, Example.prototype, "method", | |
* Object.getOwnPropertyDescriptor(Example.prototype, "method"))); | |
* | |
*/ | |
export declare function decorate(decorators: (PropertyDecorator | MethodDecorator)[], target: any, propertyKey: string | symbol, attributes?: PropertyDescriptor | null): PropertyDescriptor | undefined; | |
/** | |
* Applies a set of decorators to a property of a target object. | |
* @param decorators An array of decorators. | |
* @param target The target object. | |
* @param propertyKey The property key to decorate. | |
* @param attributes A property descriptor. | |
* @remarks Decorators are applied in reverse order. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod() { } | |
* method() { } | |
* } | |
* | |
* // property (on constructor) | |
* Reflect.decorate(decoratorsArray, Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* Reflect.decorate(decoratorsArray, Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* Object.defineProperty(Example, "staticMethod", | |
* Reflect.decorate(decoratorsArray, Example, "staticMethod", | |
* Object.getOwnPropertyDescriptor(Example, "staticMethod"))); | |
* | |
* // method (on prototype) | |
* Object.defineProperty(Example.prototype, "method", | |
* Reflect.decorate(decoratorsArray, Example.prototype, "method", | |
* Object.getOwnPropertyDescriptor(Example.prototype, "method"))); | |
* | |
*/ | |
export declare function decorate(decorators: (PropertyDecorator | MethodDecorator)[], target: any, propertyKey: string | symbol, attributes: PropertyDescriptor): PropertyDescriptor; | |
/** | |
* A default metadata decorator factory that can be used on a class, class member, or parameter. | |
* @param metadataKey The key for the metadata entry. | |
* @param metadataValue The value for the metadata entry. | |
* @returns A decorator function. | |
* @remarks | |
* If `metadataKey` is already defined for the target and target key, the | |
* metadataValue for that key will be overwritten. | |
* @example | |
* | |
* // constructor | |
* @Reflect.metadata(key, value) | |
* class Example { | |
* } | |
* | |
* // property (on constructor, TypeScript only) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* static staticProperty; | |
* } | |
* | |
* // property (on prototype, TypeScript only) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* property; | |
* } | |
* | |
* // method (on constructor) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* static staticMethod() { } | |
* } | |
* | |
* // method (on prototype) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* method() { } | |
* } | |
* | |
*/ | |
export declare function metadata(metadataKey: any, metadataValue: any): { (target: Function): void; (target: any, propertyKey: string | symbol): void; }; | |
/** | |
* Define a unique metadata entry on the target. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param metadataValue A value that contains attached metadata. | |
* @param target The target object on which to define metadata. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* Reflect.defineMetadata("custom:annotation", options, Example); | |
* | |
* // decorator factory as metadata-producing annotation. | |
* function MyAnnotation(options): ClassDecorator { | |
* return target => Reflect.defineMetadata("custom:annotation", options, target); | |
* } | |
* | |
*/ | |
export declare function defineMetadata(metadataKey: any, metadataValue: any, target: any): void; | |
/** | |
* Define a unique metadata entry on the target. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param metadataValue A value that contains attached metadata. | |
* @param target The target object on which to define metadata. | |
* @param propertyKey The property key for the target. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* Reflect.defineMetadata("custom:annotation", Number, Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* Reflect.defineMetadata("custom:annotation", Number, Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* Reflect.defineMetadata("custom:annotation", Number, Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* Reflect.defineMetadata("custom:annotation", Number, Example.prototype, "method"); | |
* | |
* // decorator factory as metadata-producing annotation. | |
* function MyAnnotation(options): PropertyDecorator { | |
* return (target, key) => Reflect.defineMetadata("custom:annotation", options, target, key); | |
* } | |
* | |
*/ | |
export declare function defineMetadata(metadataKey: any, metadataValue: any, target: any, propertyKey: string | symbol): void; | |
/** | |
* Gets a value indicating whether the target object or its prototype chain has the provided metadata key defined. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @returns `true` if the metadata key was defined on the target object or its prototype chain; otherwise, `false`. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* result = Reflect.hasMetadata("custom:annotation", Example); | |
* | |
*/ | |
export declare function hasMetadata(metadataKey: any, target: any): boolean; | |
/** | |
* Gets a value indicating whether the target object or its prototype chain has the provided metadata key defined. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey The property key for the target. | |
* @returns `true` if the metadata key was defined on the target object or its prototype chain; otherwise, `false`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* result = Reflect.hasMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.hasMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.hasMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.hasMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
export declare function hasMetadata(metadataKey: any, target: any, propertyKey: string | symbol): boolean; | |
/** | |
* Gets a value indicating whether the target object has the provided metadata key defined. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @returns `true` if the metadata key was defined on the target object; otherwise, `false`. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example); | |
* | |
*/ | |
export declare function hasOwnMetadata(metadataKey: any, target: any): boolean; | |
/** | |
* Gets a value indicating whether the target object has the provided metadata key defined. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey The property key for the target. | |
* @returns `true` if the metadata key was defined on the target object; otherwise, `false`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
export declare function hasOwnMetadata(metadataKey: any, target: any, propertyKey: string | symbol): boolean; | |
/** | |
* Gets the metadata value for the provided metadata key on the target object or its prototype chain. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @returns The metadata value for the metadata key if found; otherwise, `undefined`. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* result = Reflect.getMetadata("custom:annotation", Example); | |
* | |
*/ | |
export declare function getMetadata(metadataKey: any, target: any): any; | |
/** | |
* Gets the metadata value for the provided metadata key on the target object or its prototype chain. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey The property key for the target. | |
* @returns The metadata value for the metadata key if found; otherwise, `undefined`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* result = Reflect.getMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
export declare function getMetadata(metadataKey: any, target: any, propertyKey: string | symbol): any; | |
/** | |
* Gets the metadata value for the provided metadata key on the target object. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @returns The metadata value for the metadata key if found; otherwise, `undefined`. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* result = Reflect.getOwnMetadata("custom:annotation", Example); | |
* | |
*/ | |
export declare function getOwnMetadata(metadataKey: any, target: any): any; | |
/** | |
* Gets the metadata value for the provided metadata key on the target object. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey The property key for the target. | |
* @returns The metadata value for the metadata key if found; otherwise, `undefined`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
export declare function getOwnMetadata(metadataKey: any, target: any, propertyKey: string | symbol): any; | |
/** | |
* Gets the metadata keys defined on the target object or its prototype chain. | |
* @param target The target object on which the metadata is defined. | |
* @returns An array of unique metadata keys. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* result = Reflect.getMetadataKeys(Example); | |
* | |
*/ | |
export declare function getMetadataKeys(target: any): any[]; | |
/** | |
* Gets the metadata keys defined on the target object or its prototype chain. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey The property key for the target. | |
* @returns An array of unique metadata keys. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* result = Reflect.getMetadataKeys(Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getMetadataKeys(Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getMetadataKeys(Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getMetadataKeys(Example.prototype, "method"); | |
* | |
*/ | |
export declare function getMetadataKeys(target: any, propertyKey: string | symbol): any[]; | |
/** | |
* Gets the unique metadata keys defined on the target object. | |
* @param target The target object on which the metadata is defined. | |
* @returns An array of unique metadata keys. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* result = Reflect.getOwnMetadataKeys(Example); | |
* | |
*/ | |
export declare function getOwnMetadataKeys(target: any): any[]; | |
/** | |
* Gets the unique metadata keys defined on the target object. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey The property key for the target. | |
* @returns An array of unique metadata keys. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* result = Reflect.getOwnMetadataKeys(Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getOwnMetadataKeys(Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getOwnMetadataKeys(Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getOwnMetadataKeys(Example.prototype, "method"); | |
* | |
*/ | |
export declare function getOwnMetadataKeys(target: any, propertyKey: string | symbol): any[]; | |
/** | |
* Deletes the metadata entry from the target object with the provided key. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @returns `true` if the metadata entry was found and deleted; otherwise, false. | |
* @example | |
* | |
* class Example { | |
* } | |
* | |
* // constructor | |
* result = Reflect.deleteMetadata("custom:annotation", Example); | |
* | |
*/ | |
export declare function deleteMetadata(metadataKey: any, target: any): boolean; | |
/** | |
* Deletes the metadata entry from the target object with the provided key. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey The property key for the target. | |
* @returns `true` if the metadata entry was found and deleted; otherwise, false. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // property (on constructor) | |
* result = Reflect.deleteMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.deleteMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.deleteMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.deleteMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
export declare function deleteMetadata(metadataKey: any, target: any, propertyKey: string | symbol): boolean; | |
declare var self: { [key: string]: any }; | |
(function (this: any, factory: (exporter: <K extends keyof typeof Reflect>(key: K, value: typeof Reflect[K]) => void) => void) { | |
const root = typeof global === "object" ? global : | |
typeof self === "object" ? self : | |
typeof this === "object" ? this : | |
Function("return this;")(); | |
let exporter = makeExporter(Reflect); | |
if (typeof root.Reflect === "undefined") { | |
root.Reflect = Reflect; | |
} | |
else { | |
exporter = makeExporter(root.Reflect, exporter); | |
} | |
factory(exporter); | |
function makeExporter(target: typeof Reflect, previous?: <K extends keyof typeof Reflect>(key: K, value: typeof Reflect[K]) => void) { | |
return <K extends keyof typeof Reflect>(key: K, value: typeof Reflect[K]) => { | |
if (typeof target[key] !== "function") { | |
Object.defineProperty(target, key, { configurable: true, writable: true, value }); | |
} | |
if (previous) previous(key, value); | |
}; | |
} | |
}) | |
(function (exporter) { | |
const hasOwn = Object.prototype.hasOwnProperty; | |
// feature test for Symbol support | |
const supportsSymbol = typeof Symbol === "function"; | |
const toPrimitiveSymbol = supportsSymbol && typeof Symbol.toPrimitive !== "undefined" ? Symbol.toPrimitive : "@@toPrimitive"; | |
const iteratorSymbol = supportsSymbol && typeof Symbol.iterator !== "undefined" ? Symbol.iterator : "@@iterator"; | |
const supportsCreate = typeof Object.create === "function"; // feature test for Object.create support | |
const supportsProto = { __proto__: [] } instanceof Array; // feature test for __proto__ support | |
const downLevel = !supportsCreate && !supportsProto; | |
const HashMap = { | |
// create an object in dictionary mode (a.k.a. "slow" mode in v8) | |
create: supportsCreate | |
? <V>() => MakeDictionary(Object.create(null) as HashMap<V>) | |
: supportsProto | |
? <V>() => MakeDictionary({ __proto__: null as any } as HashMap<V>) | |
: <V>() => MakeDictionary({} as HashMap<V>), | |
has: downLevel | |
? <V>(map: HashMap<V>, key: string | number | symbol) => hasOwn.call(map, key) | |
: <V>(map: HashMap<V>, key: string | number | symbol) => key in map, | |
get: downLevel | |
? <V>(map: HashMap<V>, key: string | number | symbol): V | undefined => hasOwn.call(map, key) ? map[key as string | number] : undefined | |
: <V>(map: HashMap<V>, key: string | number | symbol): V | undefined => map[key as string | number], | |
}; | |
// Load global or shim versions of Map, Set, and WeakMap | |
const functionPrototype = Object.getPrototypeOf(Function); | |
const usePolyfill = typeof process === "object" && process.env && process.env["REFLECT_METADATA_USE_MAP_POLYFILL"] === "true"; | |
const _Map: typeof Map = !usePolyfill && typeof Map === "function" && typeof Map.prototype.entries === "function" ? Map : CreateMapPolyfill(); | |
const _Set: typeof Set = !usePolyfill && typeof Set === "function" && typeof Set.prototype.entries === "function" ? Set : CreateSetPolyfill(); | |
const _WeakMap: typeof WeakMap = !usePolyfill && typeof WeakMap === "function" ? WeakMap : CreateWeakMapPolyfill(); | |
// [[Metadata]] internal slot | |
// https://rbuckton.github.io/reflect-metadata/#ordinary-object-internal-methods-and-internal-slots | |
const Metadata = new _WeakMap<any, Map<string | symbol | undefined, Map<any, any>>>(); | |
// function decorate(decorators: ClassDecorator[], target: Function): Function; | |
// function decorate(decorators: (PropertyDecorator | MethodDecorator)[], target: any, propertyKey: string | symbol, attributes?: PropertyDescriptor | null): PropertyDescriptor | undefined; | |
// function decorate(decorators: (PropertyDecorator | MethodDecorator)[], target: any, propertyKey: string | symbol, attributes: PropertyDescriptor): PropertyDescriptor; | |
/** | |
* Applies a set of decorators to a property of a target object. | |
* @param decorators An array of decorators. | |
* @param target The target object. | |
* @param propertyKey (Optional) The property key to decorate. | |
* @param attributes (Optional) The property descriptor for the target key. | |
* @remarks Decorators are applied in reverse order. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* Example = Reflect.decorate(decoratorsArray, Example); | |
* | |
* // property (on constructor) | |
* Reflect.decorate(decoratorsArray, Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* Reflect.decorate(decoratorsArray, Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* Object.defineProperty(Example, "staticMethod", | |
* Reflect.decorate(decoratorsArray, Example, "staticMethod", | |
* Object.getOwnPropertyDescriptor(Example, "staticMethod"))); | |
* | |
* // method (on prototype) | |
* Object.defineProperty(Example.prototype, "method", | |
* Reflect.decorate(decoratorsArray, Example.prototype, "method", | |
* Object.getOwnPropertyDescriptor(Example.prototype, "method"))); | |
* | |
*/ | |
function decorate(decorators: (ClassDecorator | MemberDecorator)[], target: any, propertyKey?: string | symbol, attributes?: PropertyDescriptor | null): PropertyDescriptor | Function | undefined { | |
if (!IsUndefined(propertyKey)) { | |
if (!IsArray(decorators)) throw new TypeError(); | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsObject(attributes) && !IsUndefined(attributes) && !IsNull(attributes)) throw new TypeError(); | |
if (IsNull(attributes)) attributes = undefined; | |
propertyKey = ToPropertyKey(propertyKey); | |
return DecorateProperty(<MemberDecorator[]>decorators, target, propertyKey, attributes); | |
} | |
else { | |
if (!IsArray(decorators)) throw new TypeError(); | |
if (!IsConstructor(target)) throw new TypeError(); | |
return DecorateConstructor(<ClassDecorator[]>decorators, <Function>target); | |
} | |
} | |
exporter("decorate", <any>decorate); | |
// 4.1.2 Reflect.metadata(metadataKey, metadataValue) | |
// https://rbuckton.github.io/reflect-metadata/#reflect.metadata | |
/** | |
* A default metadata decorator factory that can be used on a class, class member, or parameter. | |
* @param metadataKey The key for the metadata entry. | |
* @param metadataValue The value for the metadata entry. | |
* @returns A decorator function. | |
* @remarks | |
* If `metadataKey` is already defined for the target and target key, the | |
* metadataValue for that key will be overwritten. | |
* @example | |
* | |
* // constructor | |
* @Reflect.metadata(key, value) | |
* class Example { | |
* } | |
* | |
* // property (on constructor, TypeScript only) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* static staticProperty; | |
* } | |
* | |
* // property (on prototype, TypeScript only) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* property; | |
* } | |
* | |
* // method (on constructor) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* static staticMethod() { } | |
* } | |
* | |
* // method (on prototype) | |
* class Example { | |
* @Reflect.metadata(key, value) | |
* method() { } | |
* } | |
* | |
*/ | |
function metadata(metadataKey: any, metadataValue: any) { | |
function decorator(target: Function): void; | |
function decorator(target: any, propertyKey: string | symbol): void; | |
function decorator(target: any, propertyKey?: string | symbol): void { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey) && !IsPropertyKey(propertyKey)) throw new TypeError(); | |
OrdinaryDefineOwnMetadata(metadataKey, metadataValue, target, propertyKey); | |
} | |
return decorator; | |
} | |
exporter("metadata", metadata); | |
// 4.1.3 Reflect.defineMetadata(metadataKey, metadataValue, target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect.definemetadata | |
function defineMetadata(metadataKey: any, metadataValue: any, target: any): void; | |
function defineMetadata(metadataKey: any, metadataValue: any, target: any, propertyKey: string | symbol): void; | |
/** | |
* Define a unique metadata entry on the target. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param metadataValue A value that contains attached metadata. | |
* @param target The target object on which to define metadata. | |
* @param propertyKey (Optional) The property key for the target. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* Reflect.defineMetadata("custom:annotation", options, Example); | |
* | |
* // property (on constructor) | |
* Reflect.defineMetadata("custom:annotation", options, Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* Reflect.defineMetadata("custom:annotation", options, Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* Reflect.defineMetadata("custom:annotation", options, Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* Reflect.defineMetadata("custom:annotation", options, Example.prototype, "method"); | |
* | |
* // decorator factory as metadata-producing annotation. | |
* function MyAnnotation(options): Decorator { | |
* return (target, key?) => Reflect.defineMetadata("custom:annotation", options, target, key); | |
* } | |
* | |
*/ | |
function defineMetadata(metadataKey: any, metadataValue: any, target: any, propertyKey?: string | symbol): void { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
return OrdinaryDefineOwnMetadata(metadataKey, metadataValue, target, propertyKey); | |
} | |
exporter("defineMetadata", defineMetadata); | |
// 4.1.4 Reflect.hasMetadata(metadataKey, target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect.hasmetadata | |
function hasMetadata(metadataKey: any, target: any): boolean; | |
function hasMetadata(metadataKey: any, target: any, propertyKey: string | symbol): boolean; | |
/** | |
* Gets a value indicating whether the target object or its prototype chain has the provided metadata key defined. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey (Optional) The property key for the target. | |
* @returns `true` if the metadata key was defined on the target object or its prototype chain; otherwise, `false`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* result = Reflect.hasMetadata("custom:annotation", Example); | |
* | |
* // property (on constructor) | |
* result = Reflect.hasMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.hasMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.hasMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.hasMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
function hasMetadata(metadataKey: any, target: any, propertyKey?: string | symbol): boolean { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
return OrdinaryHasMetadata(metadataKey, target, propertyKey); | |
} | |
exporter("hasMetadata", hasMetadata); | |
// 4.1.5 Reflect.hasOwnMetadata(metadataKey, target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect-hasownmetadata | |
function hasOwnMetadata(metadataKey: any, target: any): boolean; | |
function hasOwnMetadata(metadataKey: any, target: any, propertyKey: string | symbol): boolean; | |
/** | |
* Gets a value indicating whether the target object has the provided metadata key defined. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey (Optional) The property key for the target. | |
* @returns `true` if the metadata key was defined on the target object; otherwise, `false`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example); | |
* | |
* // property (on constructor) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.hasOwnMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
function hasOwnMetadata(metadataKey: any, target: any, propertyKey?: string | symbol): boolean { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
return OrdinaryHasOwnMetadata(metadataKey, target, propertyKey); | |
} | |
exporter("hasOwnMetadata", hasOwnMetadata); | |
// 4.1.6 Reflect.getMetadata(metadataKey, target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect-getmetadata | |
function getMetadata(metadataKey: any, target: any): any; | |
function getMetadata(metadataKey: any, target: any, propertyKey: string | symbol): any; | |
/** | |
* Gets the metadata value for the provided metadata key on the target object or its prototype chain. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey (Optional) The property key for the target. | |
* @returns The metadata value for the metadata key if found; otherwise, `undefined`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* result = Reflect.getMetadata("custom:annotation", Example); | |
* | |
* // property (on constructor) | |
* result = Reflect.getMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
function getMetadata(metadataKey: any, target: any, propertyKey?: string | symbol): any { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
return OrdinaryGetMetadata(metadataKey, target, propertyKey); | |
} | |
exporter("getMetadata", getMetadata); | |
// 4.1.7 Reflect.getOwnMetadata(metadataKey, target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect-getownmetadata | |
function getOwnMetadata(metadataKey: any, target: any): any; | |
function getOwnMetadata(metadataKey: any, target: any, propertyKey: string | symbol): any; | |
/** | |
* Gets the metadata value for the provided metadata key on the target object. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey (Optional) The property key for the target. | |
* @returns The metadata value for the metadata key if found; otherwise, `undefined`. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* result = Reflect.getOwnMetadata("custom:annotation", Example); | |
* | |
* // property (on constructor) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getOwnMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
function getOwnMetadata(metadataKey: any, target: any, propertyKey?: string | symbol): any { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
return OrdinaryGetOwnMetadata(metadataKey, target, propertyKey); | |
} | |
exporter("getOwnMetadata", getOwnMetadata); | |
// 4.1.8 Reflect.getMetadataKeys(target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect-getmetadatakeys | |
function getMetadataKeys(target: any): any[]; | |
function getMetadataKeys(target: any, propertyKey: string | symbol): any[]; | |
/** | |
* Gets the metadata keys defined on the target object or its prototype chain. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey (Optional) The property key for the target. | |
* @returns An array of unique metadata keys. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* result = Reflect.getMetadataKeys(Example); | |
* | |
* // property (on constructor) | |
* result = Reflect.getMetadataKeys(Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getMetadataKeys(Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getMetadataKeys(Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getMetadataKeys(Example.prototype, "method"); | |
* | |
*/ | |
function getMetadataKeys(target: any, propertyKey?: string | symbol): any[] { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
return OrdinaryMetadataKeys(target, propertyKey); | |
} | |
exporter("getMetadataKeys", getMetadataKeys); | |
// 4.1.9 Reflect.getOwnMetadataKeys(target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect-getownmetadata | |
function getOwnMetadataKeys(target: any): any[]; | |
function getOwnMetadataKeys(target: any, propertyKey: string | symbol): any[]; | |
/** | |
* Gets the unique metadata keys defined on the target object. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey (Optional) The property key for the target. | |
* @returns An array of unique metadata keys. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* result = Reflect.getOwnMetadataKeys(Example); | |
* | |
* // property (on constructor) | |
* result = Reflect.getOwnMetadataKeys(Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.getOwnMetadataKeys(Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.getOwnMetadataKeys(Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.getOwnMetadataKeys(Example.prototype, "method"); | |
* | |
*/ | |
function getOwnMetadataKeys(target: any, propertyKey?: string | symbol): any[] { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
return OrdinaryOwnMetadataKeys(target, propertyKey); | |
} | |
exporter("getOwnMetadataKeys", getOwnMetadataKeys); | |
// 4.1.10 Reflect.deleteMetadata(metadataKey, target [, propertyKey]) | |
// https://rbuckton.github.io/reflect-metadata/#reflect-deletemetadata | |
function deleteMetadata(metadataKey: any, target: any): boolean; | |
function deleteMetadata(metadataKey: any, target: any, propertyKey: string | symbol): boolean; | |
/** | |
* Deletes the metadata entry from the target object with the provided key. | |
* @param metadataKey A key used to store and retrieve metadata. | |
* @param target The target object on which the metadata is defined. | |
* @param propertyKey (Optional) The property key for the target. | |
* @returns `true` if the metadata entry was found and deleted; otherwise, false. | |
* @example | |
* | |
* class Example { | |
* // property declarations are not part of ES6, though they are valid in TypeScript: | |
* // static staticProperty; | |
* // property; | |
* | |
* constructor(p) { } | |
* static staticMethod(p) { } | |
* method(p) { } | |
* } | |
* | |
* // constructor | |
* result = Reflect.deleteMetadata("custom:annotation", Example); | |
* | |
* // property (on constructor) | |
* result = Reflect.deleteMetadata("custom:annotation", Example, "staticProperty"); | |
* | |
* // property (on prototype) | |
* result = Reflect.deleteMetadata("custom:annotation", Example.prototype, "property"); | |
* | |
* // method (on constructor) | |
* result = Reflect.deleteMetadata("custom:annotation", Example, "staticMethod"); | |
* | |
* // method (on prototype) | |
* result = Reflect.deleteMetadata("custom:annotation", Example.prototype, "method"); | |
* | |
*/ | |
function deleteMetadata(metadataKey: any, target: any, propertyKey?: string | symbol): boolean { | |
if (!IsObject(target)) throw new TypeError(); | |
if (!IsUndefined(propertyKey)) propertyKey = ToPropertyKey(propertyKey); | |
const metadataMap = GetOrCreateMetadataMap(target, propertyKey, /*Create*/ false); | |
if (IsUndefined(metadataMap)) return false; | |
if (!metadataMap.delete(metadataKey)) return false; | |
if (metadataMap.size > 0) return true; | |
const targetMetadata = Metadata.get(target); | |
targetMetadata.delete(propertyKey); | |
if (targetMetadata.size > 0) return true; | |
Metadata.delete(target); | |
return true; | |
} | |
exporter("deleteMetadata", deleteMetadata); | |
function DecorateConstructor(decorators: ClassDecorator[], target: Function): Function { | |
for (let i = decorators.length - 1; i >= 0; --i) { | |
const decorator = decorators[i]; | |
const decorated = decorator(target); | |
if (!IsUndefined(decorated) && !IsNull(decorated)) { | |
if (!IsConstructor(decorated)) throw new TypeError(); | |
target = <Function>decorated; | |
} | |
} | |
return target; | |
} | |
function DecorateProperty(decorators: MemberDecorator[], target: any, propertyKey: string | symbol, descriptor: PropertyDescriptor | undefined): PropertyDescriptor | undefined { | |
for (let i = decorators.length - 1; i >= 0; --i) { | |
const decorator = decorators[i]; | |
const decorated = decorator(target, propertyKey, descriptor); | |
if (!IsUndefined(decorated) && !IsNull(decorated)) { | |
if (!IsObject(decorated)) throw new TypeError(); | |
descriptor = <PropertyDescriptor>decorated; | |
} | |
} | |
return descriptor; | |
} | |
// 2.1.1 GetOrCreateMetadataMap(O, P, Create) | |
// https://rbuckton.github.io/reflect-metadata/#getorcreatemetadatamap | |
function GetOrCreateMetadataMap(O: any, P: string | symbol | undefined, Create: true): Map<any, any>; | |
function GetOrCreateMetadataMap(O: any, P: string | symbol | undefined, Create: false): Map<any, any> | undefined; | |
function GetOrCreateMetadataMap(O: any, P: string | symbol | undefined, Create: boolean): Map<any, any> | undefined { | |
let targetMetadata = Metadata.get(O); | |
if (IsUndefined(targetMetadata)) { | |
if (!Create) return undefined; | |
targetMetadata = new _Map<string | symbol | undefined, Map<any, any>>(); | |
Metadata.set(O, targetMetadata); | |
} | |
let metadataMap = targetMetadata.get(P); | |
if (IsUndefined(metadataMap)) { | |
if (!Create) return undefined; | |
metadataMap = new _Map<any, any>(); | |
targetMetadata.set(P, metadataMap); | |
} | |
return metadataMap; | |
} | |
// 3.1.1.1 OrdinaryHasMetadata(MetadataKey, O, P) | |
// https://rbuckton.github.io/reflect-metadata/#ordinaryhasmetadata | |
function OrdinaryHasMetadata(MetadataKey: any, O: any, P: string | symbol | undefined): boolean { | |
const hasOwn = OrdinaryHasOwnMetadata(MetadataKey, O, P); | |
if (hasOwn) return true; | |
const parent = OrdinaryGetPrototypeOf(O); | |
if (!IsNull(parent)) return OrdinaryHasMetadata(MetadataKey, parent, P); | |
return false; | |
} | |
// 3.1.2.1 OrdinaryHasOwnMetadata(MetadataKey, O, P) | |
// https://rbuckton.github.io/reflect-metadata/#ordinaryhasownmetadata | |
function OrdinaryHasOwnMetadata(MetadataKey: any, O: any, P: string | symbol | undefined): boolean { | |
const metadataMap = GetOrCreateMetadataMap(O, P, /*Create*/ false); | |
if (IsUndefined(metadataMap)) return false; | |
return ToBoolean(metadataMap.has(MetadataKey)); | |
} | |
// 3.1.3.1 OrdinaryGetMetadata(MetadataKey, O, P) | |
// https://rbuckton.github.io/reflect-metadata/#ordinarygetmetadata | |
function OrdinaryGetMetadata(MetadataKey: any, O: any, P: string | symbol | undefined): any { | |
const hasOwn = OrdinaryHasOwnMetadata(MetadataKey, O, P); | |
if (hasOwn) return OrdinaryGetOwnMetadata(MetadataKey, O, P); | |
const parent = OrdinaryGetPrototypeOf(O); | |
if (!IsNull(parent)) return OrdinaryGetMetadata(MetadataKey, parent, P); | |
return undefined; | |
} | |
// 3.1.4.1 OrdinaryGetOwnMetadata(MetadataKey, O, P) | |
// https://rbuckton.github.io/reflect-metadata/#ordinarygetownmetadata | |
function OrdinaryGetOwnMetadata(MetadataKey: any, O: any, P: string | symbol | undefined): any { | |
const metadataMap = GetOrCreateMetadataMap(O, P, /*Create*/ false); | |
if (IsUndefined(metadataMap)) return undefined; | |
return metadataMap.get(MetadataKey); | |
} | |
// 3.1.5.1 OrdinaryDefineOwnMetadata(MetadataKey, MetadataValue, O, P) | |
// https://rbuckton.github.io/reflect-metadata/#ordinarydefineownmetadata | |
function OrdinaryDefineOwnMetadata(MetadataKey: any, MetadataValue: any, O: any, P: string | symbol | undefined): void { | |
const metadataMap = GetOrCreateMetadataMap(O, P, /*Create*/ true); | |
metadataMap.set(MetadataKey, MetadataValue); | |
} | |
// 3.1.6.1 OrdinaryMetadataKeys(O, P) | |
// https://rbuckton.github.io/reflect-metadata/#ordinarymetadatakeys | |
function OrdinaryMetadataKeys(O: any, P: string | symbol | undefined): any[] { | |
const ownKeys = OrdinaryOwnMetadataKeys(O, P); | |
const parent = OrdinaryGetPrototypeOf(O); | |
if (parent === null) return ownKeys; | |
const parentKeys = OrdinaryMetadataKeys(parent, P); | |
if (parentKeys.length <= 0) return ownKeys; | |
if (ownKeys.length <= 0) return parentKeys; | |
const set = new _Set<any>(); | |
const keys: any[] = []; | |
for (const key of ownKeys) { | |
const hasKey = set.has(key); | |
if (!hasKey) { | |
set.add(key); | |
keys.push(key); | |
} | |
} | |
for (const key of parentKeys) { | |
const hasKey = set.has(key); | |
if (!hasKey) { | |
set.add(key); | |
keys.push(key); | |
} | |
} | |
return keys; | |
} | |
// 3.1.7.1 OrdinaryOwnMetadataKeys(O, P) | |
// https://rbuckton.github.io/reflect-metadata/#ordinaryownmetadatakeys | |
function OrdinaryOwnMetadataKeys(O: any, P: string | symbol | undefined): any[] { | |
const keys: any[] = []; | |
const metadataMap = GetOrCreateMetadataMap(O, P, /*Create*/ false); | |
if (IsUndefined(metadataMap)) return keys; | |
const keysObj = metadataMap.keys(); | |
const iterator = GetIterator(keysObj); | |
let k = 0; | |
while (true) { | |
const next = IteratorStep(iterator); | |
if (!next) { | |
keys.length = k; | |
return keys; | |
} | |
const nextValue = IteratorValue(next); | |
try { | |
keys[k] = nextValue; | |
} | |
catch (e) { | |
try { | |
IteratorClose(iterator); | |
} | |
finally { | |
throw e; | |
} | |
} | |
k++; | |
} | |
} | |
// 6 ECMAScript Data Typ0es and Values | |
// https://tc39.github.io/ecma262/#sec-ecmascript-data-types-and-values | |
function Type(x: any): Tag { | |
if (x === null) return Tag.Null; | |
switch (typeof x) { | |
case "undefined": return Tag.Undefined; | |
case "boolean": return Tag.Boolean; | |
case "string": return Tag.String; | |
case "symbol": return Tag.Symbol; | |
case "number": return Tag.Number; | |
case "object": return x === null ? Tag.Null : Tag.Object; | |
default: return Tag.Object; | |
} | |
} | |
// 6.1 ECMAScript Language Types | |
// https://tc39.github.io/ecma262/#sec-ecmascript-language-types | |
const enum Tag { | |
Undefined, | |
Null, | |
Boolean, | |
String, | |
Symbol, | |
Number, | |
Object | |
} | |
// 6.1.1 The Undefined Type | |
// https://tc39.github.io/ecma262/#sec-ecmascript-language-types-undefined-type | |
function IsUndefined(x: any): x is undefined { | |
return x === undefined; | |
} | |
// 6.1.2 The Null Type | |
// https://tc39.github.io/ecma262/#sec-ecmascript-language-types-null-type | |
function IsNull(x: any): x is null { | |
return x === null; | |
} | |
// 6.1.5 The Symbol Type | |
// https://tc39.github.io/ecma262/#sec-ecmascript-language-types-symbol-type | |
function IsSymbol(x: any): x is symbol { | |
return typeof x === "symbol"; | |
} | |
// 6.1.7 The Object Type | |
// https://tc39.github.io/ecma262/#sec-object-type | |
function IsObject<T>(x: T | undefined | null | boolean | string | symbol | number): x is T { | |
return typeof x === "object" ? x !== null : typeof x === "function"; | |
} | |
// 7.1 Type Conversion | |
// https://tc39.github.io/ecma262/#sec-type-conversion | |
// 7.1.1 ToPrimitive(input [, PreferredType]) | |
// https://tc39.github.io/ecma262/#sec-toprimitive | |
function ToPrimitive(input: any, PreferredType?: Tag): undefined | null | boolean | string | symbol | number { | |
switch (Type(input)) { | |
case Tag.Undefined: return input; | |
case Tag.Null: return input; | |
case Tag.Boolean: return input; | |
case Tag.String: return input; | |
case Tag.Symbol: return input; | |
case Tag.Number: return input; | |
} | |
const hint: "string" | "number" | "default" = PreferredType === Tag.String ? "string" : PreferredType === Tag.Number ? "number" : "default"; | |
const exoticToPrim = GetMethod(input, toPrimitiveSymbol); | |
if (exoticToPrim !== undefined) { | |
const result = exoticToPrim.call(input, hint); | |
if (IsObject(result)) throw new TypeError(); | |
return result; | |
} | |
return OrdinaryToPrimitive(input, hint === "default" ? "number" : hint); | |
} | |
// 7.1.1.1 OrdinaryToPrimitive(O, hint) | |
// https://tc39.github.io/ecma262/#sec-ordinarytoprimitive | |
function OrdinaryToPrimitive(O: any, hint: "string" | "number"): undefined | null | boolean | string | symbol | number { | |
if (hint === "string") { | |
const toString = O.toString; | |
if (IsCallable(toString)) { | |
const result = toString.call(O); | |
if (!IsObject(result)) return result; | |
} | |
const valueOf = O.valueOf; | |
if (IsCallable(valueOf)) { | |
const result = valueOf.call(O); | |
if (!IsObject(result)) return result; | |
} | |
} | |
else { | |
const valueOf = O.valueOf; | |
if (IsCallable(valueOf)) { | |
const result = valueOf.call(O); | |
if (!IsObject(result)) return result; | |
} | |
const toString = O.toString; | |
if (IsCallable(toString)) { | |
const result = toString.call(O); | |
if (!IsObject(result)) return result; | |
} | |
} | |
throw new TypeError(); | |
} | |
// 7.1.2 ToBoolean(argument) | |
// https://tc39.github.io/ecma262/2016/#sec-toboolean | |
function ToBoolean(argument: any): boolean { | |
return !!argument; | |
} | |
// 7.1.12 ToString(argument) | |
// https://tc39.github.io/ecma262/#sec-tostring | |
function ToString(argument: any): string { | |
return "" + argument; | |
} | |
// 7.1.14 ToPropertyKey(argument) | |
// https://tc39.github.io/ecma262/#sec-topropertykey | |
function ToPropertyKey(argument: any): string | symbol { | |
const key = ToPrimitive(argument, Tag.String); | |
if (IsSymbol(key)) return key; | |
return ToString(key); | |
} | |
// 7.2 Testing and Comparison Operations | |
// https://tc39.github.io/ecma262/#sec-testing-and-comparison-operations | |
// 7.2.2 IsArray(argument) | |
// https://tc39.github.io/ecma262/#sec-isarray | |
function IsArray(argument: any): argument is any[] { | |
return Array.isArray | |
? Array.isArray(argument) | |
: argument instanceof Object | |
? argument instanceof Array | |
: Object.prototype.toString.call(argument) === "[object Array]"; | |
} | |
// 7.2.3 IsCallable(argument) | |
// https://tc39.github.io/ecma262/#sec-iscallable | |
function IsCallable(argument: any): argument is Function { | |
// NOTE: This is an approximation as we cannot check for [[Call]] internal method. | |
return typeof argument === "function"; | |
} | |
// 7.2.4 IsConstructor(argument) | |
// https://tc39.github.io/ecma262/#sec-isconstructor | |
function IsConstructor(argument: any): argument is Function { | |
// NOTE: This is an approximation as we cannot check for [[Construct]] internal method. | |
return typeof argument === "function"; | |
} | |
// 7.2.7 IsPropertyKey(argument) | |
// https://tc39.github.io/ecma262/#sec-ispropertykey | |
function IsPropertyKey(argument: any): argument is string | symbol { | |
switch (Type(argument)) { | |
case Tag.String: return true; | |
case Tag.Symbol: return true; | |
default: return false; | |
} | |
} | |
// 7.3 Operations on Objects | |
// https://tc39.github.io/ecma262/#sec-operations-on-objects | |
// 7.3.9 GetMethod(V, P) | |
// https://tc39.github.io/ecma262/#sec-getmethod | |
function GetMethod(V: any, P: any): Function | undefined { | |
const func = V[P]; | |
if (func === undefined || func === null) return undefined; | |
if (!IsCallable(func)) throw new TypeError(); | |
return func; | |
} | |
// 7.4 Operations on Iterator Objects | |
// https://tc39.github.io/ecma262/#sec-operations-on-iterator-objects | |
function GetIterator<T>(obj: Iterable<T>): Iterator<T> { | |
const method = GetMethod(obj, iteratorSymbol); | |
if (!IsCallable(method)) throw new TypeError(); // from Call | |
const iterator = method.call(obj); | |
if (!IsObject(iterator)) throw new TypeError(); | |
return iterator; | |
} | |
// 7.4.4 IteratorValue(iterResult) | |
// https://tc39.github.io/ecma262/2016/#sec-iteratorvalue | |
function IteratorValue<T>(iterResult: IteratorResult<T>): T { | |
return iterResult.value; | |
} | |
// 7.4.5 IteratorStep(iterator) | |
// https://tc39.github.io/ecma262/#sec-iteratorstep | |
function IteratorStep<T>(iterator: Iterator<T>): IteratorResult<T> | false { | |
const result = iterator.next(); | |
return result.done ? false : result; | |
} | |
// 7.4.6 IteratorClose(iterator, completion) | |
// https://tc39.github.io/ecma262/#sec-iteratorclose | |
function IteratorClose<T>(iterator: Iterator<T>) { | |
const f = iterator["return"]; | |
if (f) f.call(iterator); | |
} | |
// 9.1 Ordinary Object Internal Methods and Internal Slots | |
// https://tc39.github.io/ecma262/#sec-ordinary-object-internal-methods-and-internal-slots | |
// 9.1.1.1 OrdinaryGetPrototypeOf(O) | |
// https://tc39.github.io/ecma262/#sec-ordinarygetprototypeof | |
function OrdinaryGetPrototypeOf(O: any): any { | |
const proto = Object.getPrototypeOf(O); | |
if (typeof O !== "function" || O === functionPrototype) return proto; | |
// TypeScript doesn't set __proto__ in ES5, as it's non-standard. | |
// Try to determine the superclass constructor. Compatible implementations | |
// must either set __proto__ on a subclass constructor to the superclass constructor, | |
// or ensure each class has a valid `constructor` property on its prototype that | |
// points back to the constructor. | |
// If this is not the same as Function.[[Prototype]], then this is definately inherited. | |
// This is the case when in ES6 or when using __proto__ in a compatible browser. | |
if (proto !== functionPrototype) return proto; | |
// If the super prototype is Object.prototype, null, or undefined, then we cannot determine the heritage. | |
const prototype = O.prototype; | |
const prototypeProto = prototype && Object.getPrototypeOf(prototype); | |
if (prototypeProto == null || prototypeProto === Object.prototype) return proto; | |
// If the constructor was not a function, then we cannot determine the heritage. | |
const constructor = prototypeProto.constructor; | |
if (typeof constructor !== "function") return proto; | |
// If we have some kind of self-reference, then we cannot determine the heritage. | |
if (constructor === O) return proto; | |
// we have a pretty good guess at the heritage. | |
return constructor; | |
} | |
// naive Map shim | |
function CreateMapPolyfill(): MapConstructor { | |
const cacheSentinel = {}; | |
const arraySentinel: any[] = []; | |
class MapIterator<K, V, R extends (K | V | [K, V])> implements IterableIterator<R> { | |
private _keys: K[]; | |
private _values: V[]; | |
private _index = 0; | |
private _selector: (key: K, value: V) => R; | |
constructor(keys: K[], values: V[], selector: (key: K, value: V) => R) { | |
this._keys = keys; | |
this._values = values; | |
this._selector = selector; | |
} | |
"@@iterator"() { return this; } | |
[iteratorSymbol]() { return this; } | |
next(): IteratorResult<R> { | |
const index = this._index; | |
if (index >= 0 && index < this._keys.length) { | |
const result = this._selector(this._keys[index], this._values[index]); | |
if (index + 1 >= this._keys.length) { | |
this._index = -1; | |
this._keys = arraySentinel; | |
this._values = arraySentinel; | |
} | |
else { | |
this._index++; | |
} | |
return { value: result, done: false }; | |
} | |
return { value: <never>undefined, done: true }; | |
} | |
throw(error: any): IteratorResult<R> { | |
if (this._index >= 0) { | |
this._index = -1; | |
this._keys = arraySentinel; | |
this._values = arraySentinel; | |
} | |
throw error; | |
} | |
return(value?: R): IteratorResult<R> { | |
if (this._index >= 0) { | |
this._index = -1; | |
this._keys = arraySentinel; | |
this._values = arraySentinel; | |
} | |
return { value: <never>value, done: true }; | |
} | |
} | |
return class Map<K, V> { | |
private _keys: K[] = []; | |
private _values: (V | undefined)[] = []; | |
private _cacheKey = cacheSentinel; | |
private _cacheIndex = -2; | |
get size() { return this._keys.length; } | |
has(key: K): boolean { return this._find(key, /*insert*/ false) >= 0; } | |
get(key: K): V | undefined { | |
const index = this._find(key, /*insert*/ false); | |
return index >= 0 ? this._values[index] : undefined; | |
} | |
set(key: K, value: V): this { | |
const index = this._find(key, /*insert*/ true); | |
this._values[index] = value; | |
return this; | |
} | |
delete(key: K): boolean { | |
const index = this._find(key, /*insert*/ false); | |
if (index >= 0) { | |
const size = this._keys.length; | |
for (let i = index + 1; i < size; i++) { | |
this._keys[i - 1] = this._keys[i]; | |
this._values[i - 1] = this._values[i]; | |
} | |
this._keys.length--; | |
this._values.length--; | |
if (key === this._cacheKey) { | |
this._cacheKey = cacheSentinel; | |
this._cacheIndex = -2; | |
} | |
return true; | |
} | |
return false; | |
} | |
clear(): void { | |
this._keys.length = 0; | |
this._values.length = 0; | |
this._cacheKey = cacheSentinel; | |
this._cacheIndex = -2; | |
} | |
keys() { return new MapIterator(this._keys, this._values, getKey); } | |
values() { return new MapIterator(this._keys, this._values, getValue); } | |
entries() { return new MapIterator(this._keys, this._values, getEntry); } | |
"@@iterator"() { return this.entries(); } | |
[iteratorSymbol]() { return this.entries(); } | |
private _find(key: K, insert?: boolean): number { | |
if (this._cacheKey !== key) { | |
this._cacheIndex = this._keys.indexOf(this._cacheKey = key); | |
} | |
if (this._cacheIndex < 0 && insert) { | |
this._cacheIndex = this._keys.length; | |
this._keys.push(key); | |
this._values.push(undefined); | |
} | |
return this._cacheIndex; | |
} | |
}; | |
function getKey<K, V>(key: K, _: V) { | |
return key; | |
} | |
function getValue<K, V>(_: K, value: V) { | |
return value; | |
} | |
function getEntry<K, V>(key: K, value: V) { | |
return [key, value] as [K, V]; | |
} | |
} | |
// naive Set shim | |
function CreateSetPolyfill(): SetConstructor { | |
return class Set<T> { | |
private _map = new _Map<any, any>(); | |
get size() { return this._map.size; } | |
has(value: T): boolean { return this._map.has(value); } | |
add(value: T): Set<T> { return this._map.set(value, value), this; } | |
delete(value: T): boolean { return this._map.delete(value); } | |
clear(): void { this._map.clear(); } | |
keys() { return this._map.keys(); } | |
values() { return this._map.values(); } | |
entries() { return this._map.entries(); } | |
"@@iterator"() { return this.keys(); } | |
[iteratorSymbol]() { return this.keys(); } | |
}; | |
} | |
// naive WeakMap shim | |
function CreateWeakMapPolyfill(): WeakMapConstructor { | |
const UUID_SIZE = 16; | |
const keys = HashMap.create<boolean>(); | |
const rootKey = CreateUniqueKey(); | |
return class WeakMap<K, V> { | |
private _key = CreateUniqueKey(); | |
has(target: K): boolean { | |
const table = GetOrCreateWeakMapTable<K>(target, /*create*/ false); | |
return table !== undefined ? HashMap.has(table, this._key) : false; | |
} | |
get(target: K): V { | |
const table = GetOrCreateWeakMapTable<K>(target, /*create*/ false); | |
return table !== undefined ? HashMap.get(table, this._key) : undefined; | |
} | |
set(target: K, value: V): WeakMap<K, V> { | |
const table = GetOrCreateWeakMapTable<K>(target, /*create*/ true); | |
table[this._key] = value; | |
return this; | |
} | |
delete(target: K): boolean { | |
const table = GetOrCreateWeakMapTable<K>(target, /*create*/ false); | |
return table !== undefined ? delete table[this._key] : false; | |
} | |
clear(): void { | |
// NOTE: not a real clear, just makes the previous data unreachable | |
this._key = CreateUniqueKey(); | |
} | |
}; | |
function CreateUniqueKey(): string { | |
let key: string; | |
do key = "@@WeakMap@@" + CreateUUID(); | |
while (HashMap.has(keys, key)); | |
keys[key] = true; | |
return key; | |
} | |
function GetOrCreateWeakMapTable<K>(target: K, create: true): HashMap<any>; | |
function GetOrCreateWeakMapTable<K>(target: K, create: false): HashMap<any> | undefined; | |
function GetOrCreateWeakMapTable<K>(target: K, create: boolean): HashMap<any> | undefined { | |
if (!hasOwn.call(target, rootKey)) { | |
if (!create) return undefined; | |
Object.defineProperty(target, rootKey, { value: HashMap.create<any>() }); | |
} | |
return (<any>target)[rootKey]; | |
} | |
function FillRandomBytes(buffer: BufferLike, size: number): BufferLike { | |
for (let i = 0; i < size; ++i) buffer[i] = Math.random() * 0xff | 0; | |
return buffer; | |
} | |
function GenRandomBytes(size: number): BufferLike { | |
if (typeof Uint8Array === "function") { | |
if (typeof crypto !== "undefined") return crypto.getRandomValues(new Uint8Array(size)) as Uint8Array; | |
if (typeof msCrypto !== "undefined") return msCrypto.getRandomValues(new Uint8Array(size)) as Uint8Array; | |
return FillRandomBytes(new Uint8Array(size), size); | |
} | |
return FillRandomBytes(new Array(size), size); | |
} | |
function CreateUUID() { | |
const data = GenRandomBytes(UUID_SIZE); | |
// mark as random - RFC 4122 § 4.4 | |
data[6] = data[6] & 0x4f | 0x40; | |
data[8] = data[8] & 0xbf | 0x80; | |
let result = ""; | |
for (let offset = 0; offset < UUID_SIZE; ++offset) { | |
const byte = data[offset]; | |
if (offset === 4 || offset === 6 || offset === 8) result += "-"; | |
if (byte < 16) result += "0"; | |
result += byte.toString(16).toLowerCase(); | |
} | |
return result; | |
} | |
} | |
// uses a heuristic used by v8 and chakra to force an object into dictionary mode. | |
function MakeDictionary<T>(obj: T): T { | |
(<any>obj).__ = undefined; | |
delete (<any>obj).__; | |
return obj; | |
} | |
}); | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment