Socket
Socket
Sign inDemoInstall

@glimmer/validator

Package Overview
Dependencies
Maintainers
12
Versions
143
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@glimmer/validator - npm Package Compare versions

Comparing version 0.55.1 to 0.55.2

2

dist/commonjs/es2017/lib/validators.js

@@ -291,2 +291,2 @@ "use strict";

}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { UnionToIntersection, symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\ninterface MonomorphicTagMapping {\n  [MonomorphicTagTypes.Dirtyable]: DirtyableTag;\n  [MonomorphicTagTypes.Updatable]: UpdatableTag;\n  [MonomorphicTagTypes.Combinator]: CombinatorTag;\n  [MonomorphicTagTypes.Constant]: ConstantTag;\n}\n\ntype MonomorphicTag = UnionToIntersection<MonomorphicTagMapping[MonomorphicTagTypes]>;\ntype MonomorphicTagType = UnionToIntersection<MonomorphicTagTypes>;\n\nclass MonomorphicTagImpl implements MonomorphicTag {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: MonomorphicTagType;\n\n  constructor(type: MonomorphicTagTypes) {\n    this[TYPE] = type as MonomorphicTagType;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG = new MonomorphicTagImpl(MonomorphicTagTypes.Constant) as ConstantTag;\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator) as CombinatorTag;\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\nclass MonomorphicTagImpl<T extends MonomorphicTagTypes = MonomorphicTagTypes> {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: T;\n\n  constructor(type: T) {\n    this[TYPE] = type;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG: ConstantTag = new MonomorphicTagImpl(MonomorphicTagTypes.Constant);\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag: CombinatorTag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator);\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}

@@ -299,2 +299,2 @@ "use strict";

}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { UnionToIntersection, symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\ninterface MonomorphicTagMapping {\n  [MonomorphicTagTypes.Dirtyable]: DirtyableTag;\n  [MonomorphicTagTypes.Updatable]: UpdatableTag;\n  [MonomorphicTagTypes.Combinator]: CombinatorTag;\n  [MonomorphicTagTypes.Constant]: ConstantTag;\n}\n\ntype MonomorphicTag = UnionToIntersection<MonomorphicTagMapping[MonomorphicTagTypes]>;\ntype MonomorphicTagType = UnionToIntersection<MonomorphicTagTypes>;\n\nclass MonomorphicTagImpl implements MonomorphicTag {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: MonomorphicTagType;\n\n  constructor(type: MonomorphicTagTypes) {\n    this[TYPE] = type as MonomorphicTagType;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG = new MonomorphicTagImpl(MonomorphicTagTypes.Constant) as ConstantTag;\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator) as CombinatorTag;\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\nclass MonomorphicTagImpl<T extends MonomorphicTagTypes = MonomorphicTagTypes> {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: T;\n\n  constructor(type: T) {\n    this[TYPE] = type;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG: ConstantTag = new MonomorphicTagImpl(MonomorphicTagTypes.Constant);\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag: CombinatorTag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator);\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}

@@ -244,2 +244,2 @@ import { DEBUG } from '@glimmer/env';

}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { UnionToIntersection, symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\ninterface MonomorphicTagMapping {\n  [MonomorphicTagTypes.Dirtyable]: DirtyableTag;\n  [MonomorphicTagTypes.Updatable]: UpdatableTag;\n  [MonomorphicTagTypes.Combinator]: CombinatorTag;\n  [MonomorphicTagTypes.Constant]: ConstantTag;\n}\n\ntype MonomorphicTag = UnionToIntersection<MonomorphicTagMapping[MonomorphicTagTypes]>;\ntype MonomorphicTagType = UnionToIntersection<MonomorphicTagTypes>;\n\nclass MonomorphicTagImpl implements MonomorphicTag {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: MonomorphicTagType;\n\n  constructor(type: MonomorphicTagTypes) {\n    this[TYPE] = type as MonomorphicTagType;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG = new MonomorphicTagImpl(MonomorphicTagTypes.Constant) as ConstantTag;\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator) as CombinatorTag;\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\nclass MonomorphicTagImpl<T extends MonomorphicTagTypes = MonomorphicTagTypes> {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: T;\n\n  constructor(type: T) {\n    this[TYPE] = type;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG: ConstantTag = new MonomorphicTagImpl(MonomorphicTagTypes.Constant);\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag: CombinatorTag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator);\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}

@@ -252,2 +252,2 @@ import { DEBUG } from '@glimmer/env';

}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { UnionToIntersection, symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\ninterface MonomorphicTagMapping {\n  [MonomorphicTagTypes.Dirtyable]: DirtyableTag;\n  [MonomorphicTagTypes.Updatable]: UpdatableTag;\n  [MonomorphicTagTypes.Combinator]: CombinatorTag;\n  [MonomorphicTagTypes.Constant]: ConstantTag;\n}\n\ntype MonomorphicTag = UnionToIntersection<MonomorphicTagMapping[MonomorphicTagTypes]>;\ntype MonomorphicTagType = UnionToIntersection<MonomorphicTagTypes>;\n\nclass MonomorphicTagImpl implements MonomorphicTag {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: MonomorphicTagType;\n\n  constructor(type: MonomorphicTagTypes) {\n    this[TYPE] = type as MonomorphicTagType;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG = new MonomorphicTagImpl(MonomorphicTagTypes.Constant) as ConstantTag;\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator) as CombinatorTag;\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/validator/lib/validators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DEBUG } from '@glimmer/env';\nimport { symbol } from './utils';\nimport { assertTagNotConsumed } from './debug';\n\n//////////\n\nexport type Revision = number;\n\nexport const CONSTANT: Revision = 0;\nexport const INITIAL: Revision = 1;\nexport const VOLATILE: Revision = 9007199254740991; // MAX_INT\n\nlet $REVISION = INITIAL;\n\nexport function bump() {\n  $REVISION++;\n}\n\n//////////\n\nexport const COMPUTE: unique symbol = symbol('TAG_COMPUTE');\n\nexport interface EntityTag<T> {\n  [COMPUTE](): T;\n}\n\nexport interface Tag extends EntityTag<Revision> {}\n\nexport interface EntityTagged<T> {\n  tag: EntityTag<T>;\n}\n\nexport interface Tagged {\n  tag: Tag;\n}\n\n//////////\n\n/**\n * `value` receives a tag and returns an opaque Revision based on that tag. This\n * snapshot can then later be passed to `validate` with the same tag to\n * determine if the tag has changed at all since the time that `value` was\n * called.\n *\n * The current implementation returns the global revision count directly for\n * performance reasons. This is an implementation detail, and should not be\n * relied on directly by users of these APIs. Instead, Revisions should be\n * treated as if they are opaque/unknown, and should only be interacted with via\n * the `value`/`validate` API.\n *\n * @param tag\n */\nexport function valueForTag(_tag: Tag): Revision {\n  return $REVISION;\n}\n\n/**\n * `validate` receives a tag and a snapshot from a previous call to `value` with\n * the same tag, and determines if the tag is still valid compared to the\n * snapshot. If the tag's state has changed at all since then, `validate` will\n * return false, otherwise it will return true. This is used to determine if a\n * calculation related to the tags should be rerun.\n *\n * @param tag\n * @param snapshot\n */\nexport function validateTag(tag: Tag, snapshot: Revision) {\n  return snapshot >= tag[COMPUTE]();\n}\n\n//////////\n\n/**\n * This enum represents all of the possible tag types for the monomorphic tag class.\n * Other custom tag classes can exist, such as CurrentTag and VolatileTag, but for\n * performance reasons, any type of tag that is meant to be used frequently should\n * be added to the monomorphic tag.\n */\nconst enum MonomorphicTagTypes {\n  Dirtyable,\n  Updatable,\n  Combinator,\n  Constant,\n}\n\nconst TYPE: unique symbol = symbol('TAG_TYPE');\n\nexport let ALLOW_CYCLES: WeakMap<Tag, boolean> | undefined;\n\nif (DEBUG) {\n  ALLOW_CYCLES = new WeakMap();\n}\n\ninterface MonomorphicTagBase<T extends MonomorphicTagTypes> extends Tag {\n  [TYPE]: T;\n}\n\nexport interface DirtyableTag extends MonomorphicTagBase<MonomorphicTagTypes.Dirtyable> {}\nexport interface UpdatableTag extends MonomorphicTagBase<MonomorphicTagTypes.Updatable> {}\nexport interface CombinatorTag extends MonomorphicTagBase<MonomorphicTagTypes.Combinator> {}\nexport interface ConstantTag extends MonomorphicTagBase<MonomorphicTagTypes.Constant> {}\n\nclass MonomorphicTagImpl<T extends MonomorphicTagTypes = MonomorphicTagTypes> {\n  private revision = INITIAL;\n  private lastChecked = INITIAL;\n  private lastValue = INITIAL;\n\n  private isUpdating = false;\n  private subtags: Tag[] | null = null;\n\n  private subtag: Tag | null = null;\n  private subtagBufferCache: Revision | null = null;\n\n  [TYPE]: T;\n\n  constructor(type: T) {\n    this[TYPE] = type;\n  }\n\n  [COMPUTE](): Revision {\n    let { lastChecked } = this;\n\n    if (this.isUpdating === true) {\n      if (DEBUG && !ALLOW_CYCLES!.has(this)) {\n        throw new Error('Cycles in tags are not allowed');\n      }\n\n      this.lastChecked = ++$REVISION;\n    } else if (lastChecked !== $REVISION) {\n      this.isUpdating = true;\n      this.lastChecked = $REVISION;\n\n      try {\n        let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;\n\n        if (subtag !== null) {\n          let subtagValue = subtag[COMPUTE]();\n\n          if (subtagValue === subtagBufferCache) {\n            revision = Math.max(revision, lastValue);\n          } else {\n            // Clear the temporary buffer cache\n            this.subtagBufferCache = null;\n            revision = Math.max(revision, subtagValue);\n          }\n        }\n\n        if (subtags !== null) {\n          for (let i = 0; i < subtags.length; i++) {\n            let value = subtags[i][COMPUTE]();\n            revision = Math.max(value, revision);\n          }\n        }\n\n        this.lastValue = revision;\n      } finally {\n        this.isUpdating = false;\n      }\n    }\n\n    return this.lastValue;\n  }\n\n  static updateTag(_tag: UpdatableTag, _subtag: Tag) {\n    if (DEBUG && _tag[TYPE] !== MonomorphicTagTypes.Updatable) {\n      throw new Error('Attempted to update a tag that was not updatable');\n    }\n\n    // TODO: TS 3.7 should allow us to do this via assertion\n    let tag = _tag as MonomorphicTagImpl;\n    let subtag = _subtag as MonomorphicTagImpl;\n\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      // There are two different possibilities when updating a subtag:\n      //\n      // 1. subtag[COMPUTE]() <= tag[COMPUTE]();\n      // 2. subtag[COMPUTE]() > tag[COMPUTE]();\n      //\n      // The first possibility is completely fine within our caching model, but\n      // the second possibility presents a problem. If the parent tag has\n      // already been read, then it's value is cached and will not update to\n      // reflect the subtag's greater value. Next time the cache is busted, the\n      // subtag's value _will_ be read, and it's value will be _greater_ than\n      // the saved snapshot of the parent, causing the resulting calculation to\n      // be rerun erroneously.\n      //\n      // In order to prevent this, when we first update to a new subtag we store\n      // its computed value, and then check against that computed value on\n      // subsequent updates. If its value hasn't changed, then we return the\n      // parent's previous value. Once the subtag changes for the first time,\n      // we clear the cache and everything is finally in sync with the parent.\n      tag.subtagBufferCache = subtag[COMPUTE]();\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirtyTag(tag: DirtyableTag | UpdatableTag) {\n    if (\n      DEBUG &&\n      !(tag[TYPE] === MonomorphicTagTypes.Updatable || tag[TYPE] === MonomorphicTagTypes.Dirtyable)\n    ) {\n      throw new Error('Attempted to dirty a tag that was not dirtyable');\n    }\n\n    if (DEBUG) {\n      // Usually by this point, we've already asserted with better error information,\n      // but this is our last line of defense.\n      assertTagNotConsumed!(tag);\n    }\n\n    (tag as MonomorphicTagImpl).revision = ++$REVISION;\n  }\n}\n\nexport const dirtyTag = MonomorphicTagImpl.dirtyTag;\nexport const updateTag = MonomorphicTagImpl.updateTag;\n\n//////////\n\nexport function createTag(): DirtyableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Dirtyable);\n}\n\nexport function createUpdatableTag(): UpdatableTag {\n  return new MonomorphicTagImpl(MonomorphicTagTypes.Updatable);\n}\n\n//////////\n\nexport const CONSTANT_TAG: ConstantTag = new MonomorphicTagImpl(MonomorphicTagTypes.Constant);\n\nexport function isConstTagged({ tag }: Tagged): boolean {\n  return tag === CONSTANT_TAG;\n}\n\nexport function isConstTag(tag: Tag): tag is ConstantTag {\n  return tag === CONSTANT_TAG;\n}\n\n//////////\n\nexport class VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nexport class CurrentTag implements CurrentTag {\n  [COMPUTE]() {\n    return $REVISION;\n  }\n}\n\nexport const CURRENT_TAG = new CurrentTag();\n\n//////////\n\nexport function combine(tags: Tag[]): Tag {\n  let optimized: Tag[] = [];\n\n  for (let i = 0, l = tags.length; i < l; i++) {\n    let tag = tags[i];\n    if (tag === CONSTANT_TAG) continue;\n    optimized.push(tag);\n  }\n\n  return createCombinatorTag(optimized);\n}\n\nexport function createCombinatorTag(tags: Tag[]): Tag {\n  switch (tags.length) {\n    case 0:\n      return CONSTANT_TAG;\n    case 1:\n      return tags[0];\n    default:\n      let tag: CombinatorTag = new MonomorphicTagImpl(MonomorphicTagTypes.Combinator);\n      (tag as any).subtags = tags;\n      return tag;\n  }\n}\n"],"sourceRoot":""}

@@ -1,2 +0,1 @@

import { UnionToIntersection } from './utils';
export declare type Revision = number;

@@ -70,11 +69,3 @@ export declare const CONSTANT: Revision;

}
interface MonomorphicTagMapping {
[MonomorphicTagTypes.Dirtyable]: DirtyableTag;
[MonomorphicTagTypes.Updatable]: UpdatableTag;
[MonomorphicTagTypes.Combinator]: CombinatorTag;
[MonomorphicTagTypes.Constant]: ConstantTag;
}
declare type MonomorphicTag = UnionToIntersection<MonomorphicTagMapping[MonomorphicTagTypes]>;
declare type MonomorphicTagType = UnionToIntersection<MonomorphicTagTypes>;
declare class MonomorphicTagImpl implements MonomorphicTag {
declare class MonomorphicTagImpl<T extends MonomorphicTagTypes = MonomorphicTagTypes> {
private revision;

@@ -87,4 +78,4 @@ private lastChecked;

private subtagBufferCache;
[TYPE]: MonomorphicTagType;
constructor(type: MonomorphicTagTypes);
[TYPE]: T;
constructor(type: T);
[COMPUTE](): Revision;

@@ -91,0 +82,0 @@ static updateTag(_tag: UpdatableTag, _subtag: Tag): void;

{
"name": "@glimmer/validator",
"version": "0.55.1",
"version": "0.55.2",
"description": "Objects used to track values and their dirtiness in Glimmer",

@@ -5,0 +5,0 @@ "license": "MIT",

Sorry, the diff of this file is too big to display

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

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc