Socket
Socket
Sign inDemoInstall

@glimmer/validator

Package Overview
Dependencies
Maintainers
11
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.46.0 to 0.47.0

77

dist/commonjs/es2017/lib/validators.js

@@ -47,8 +47,3 @@ 'use strict';

*/
function value(tag) {
if (_env.DEBUG) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
function value(_tag) {
return $REVISION;

@@ -67,17 +62,4 @@ }

function validate(tag, snapshot) {
if (_env.DEBUG) {
IS_VALIDATING = true;
}
let isValid = snapshot >= tag[COMPUTE]();
if (_env.DEBUG) {
IS_VALIDATING = false;
if (isValid) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
}
return isValid;
return snapshot >= tag[COMPUTE]();
}
let IS_VALIDATING;
const TYPE = (0, _utils.symbol)('TAG_TYPE');

@@ -94,4 +76,5 @@ let ALLOW_CYCLES = exports.ALLOW_CYCLES = undefined;

this.isUpdating = false;
this.subtags = null;
this.subtag = null;
this.subtags = null;
this.subtagBufferCache = null;
this[TYPE] = type;

@@ -108,19 +91,14 @@ }

this.isUpdating = true;
if (_env.DEBUG) {
// In DEBUG, we don't cache while validating only, because it is valid
// update a tag between calling `validate()` and `value()`. Once you
// call `value()` on a tag, its revision is effectively locked in, and
// if you attempt to update it to a tag that is more recent it could
// break assumptions in our system. This is why the assertion exists in
// the static `update()` method below.
if (!IS_VALIDATING) {
this.lastChecked = $REVISION;
}
} else {
this.lastChecked = $REVISION;
}
this.lastChecked = $REVISION;
try {
let { subtags, subtag, revision } = this;
let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;
if (subtag !== null) {
revision = Math.max(revision, subtag[COMPUTE]());
let subtagValue = subtag[COMPUTE]();
if (subtagValue === subtagBufferCache) {
revision = Math.max(revision, lastValue);
} else {
// Clear the temporary buffer cache
this.subtagBufferCache = null;
revision = Math.max(revision, subtagValue);
}
}

@@ -140,3 +118,3 @@ if (subtags !== null) {

}
static update(_tag, subtag) {
static update(_tag, _subtag) {
if (_env.DEBUG && _tag[TYPE] !== 1 /* Updatable */) {

@@ -147,8 +125,25 @@ throw new Error('Attempted to update a tag that was not updatable');

let tag = _tag;
let subtag = _subtag;
if (subtag === CONSTANT_TAG) {
tag.subtag = null;
} else {
if (_env.DEBUG && tag.lastChecked === $REVISION && subtag[COMPUTE]() > tag.lastValue) {
throw new Error('BUG: attempted to update a tag with a tag that has a more recent revision as its value');
}
// There are two different possibilities when updating a subtag:
//
// 1. subtag[COMPUTE]() <= tag[COMPUTE]();
// 2. subtag[COMPUTE]() > tag[COMPUTE]();
//
// The first possibility is completely fine within our caching model, but
// the second possibility presents a problem. If the parent tag has
// already been read, then it's value is cached and will not update to
// reflect the subtag's greater value. Next time the cache is busted, the
// subtag's value _will_ be read, and it's value will be _greater_ than
// the saved snapshot of the parent, causing the resulting calculation to
// be rerun erroneously.
//
// In order to prevent this, when we first update to a new subtag we store
// its computed value, and then check against that computed value on
// subsequent updates. If its value hasn't changed, then we return the
// parent's previous value. Once the subtag changes for the first time,
// we clear the cache and everything is finally in sync with the parent.
tag.subtagBufferCache = subtag[COMPUTE]();
tag.subtag = subtag;

@@ -222,2 +217,2 @@ }

}
//# 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 value(tag: Tag): Revision {\n  if (DEBUG) {\n    // compute to cache the latest value, which will prevent us from doing\n    // invalid updates later on.\n    tag[COMPUTE]();\n  }\n\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 validate(tag: Tag, snapshot: Revision) {\n  if (DEBUG) {\n    IS_VALIDATING = true;\n  }\n\n  let isValid = snapshot >= tag[COMPUTE]();\n\n  if (DEBUG) {\n    IS_VALIDATING = false;\n\n    if (isValid) {\n      // compute to cache the latest value, which will prevent us from doing\n      // invalid updates later on.\n      tag[COMPUTE]();\n    }\n  }\n\n  return isValid;\n}\n\nlet IS_VALIDATING: boolean | undefined;\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 subtag: Tag | null = null;\n  private subtags: Tag[] | 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\n      if (DEBUG) {\n        // In DEBUG, we don't cache while validating only, because it is valid\n        // update a tag between calling `validate()` and `value()`. Once you\n        // call `value()` on a tag, its revision is effectively locked in, and\n        // if you attempt to update it to a tag that is more recent it could\n        // break assumptions in our system. This is why the assertion exists in\n        // the static `update()` method below.\n        if (!IS_VALIDATING) {\n          this.lastChecked = $REVISION;\n        }\n      } else {\n        this.lastChecked = $REVISION;\n      }\n\n      try {\n        let { subtags, subtag, revision } = this;\n\n        if (subtag !== null) {\n          revision = Math.max(revision, subtag[COMPUTE]());\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 update(_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\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      if (\n        DEBUG &&\n        tag.lastChecked === $REVISION &&\n        (subtag as MonomorphicTagImpl)[COMPUTE]() > tag.lastValue\n      ) {\n        throw new Error(\n          'BUG: attempted to update a tag with a tag that has a more recent revision as its value'\n        );\n      }\n\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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":";;;;;;QAcM,I,GAAA,I;QAsCA,K,GAAA,K;QAcA,Q,GAAA,Q;QAqKA,S,GAAA,S;QAIA,kB,GAAA,kB;QAQA,O,GAAA,O;QAIA,U,GAAA,U;QA0BA,O,GAAA,O;QAYA,mB,GAAA,mB;;;;AA5RN;;AACA;;AAMO,MAAM,8BAAN,CAAA;AACA,MAAM,4BAAN,CAAA;AACA,MAAM,8BAAN,gBAAA,C,CAA6C;AAEpD,IAAI,YAAJ,OAAA;AAEM,SAAA,IAAA,GAAc;AAClB;AACD;AAED;AAEO,MAAM,4BAAyB,mBAA/B,aAA+B,CAA/B;AAgBP;AAEA;;;;;;;;;;;;;;AAcM,SAAA,KAAA,CAAA,IAAA,EAAyB;AAC7B,WAAA,SAAA;AACD;AAED;;;;;;;;;;AAUM,SAAA,QAAA,CAAA,GAAA,EAAA,QAAA,EAA+C;AACnD,WAAO,YAAY,IAAnB,OAAmB,GAAnB;AACD;AAiBD,MAAM,OAAsB,mBAA5B,UAA4B,CAA5B;AAEO,IAAA,+CAAA;AAEP,IAAA,UAAA,EAAW;AACT,YAHK,YAGL,kBAAe,IAAf,OAAe,EAAf;AACD;AAqBD,MAAA,kBAAA,CAAwB;AAatB,gBAAA,IAAA,EAAqC;AAZ7B,aAAA,QAAA,GAAA,OAAA;AACA,aAAA,WAAA,GAAA,OAAA;AACA,aAAA,SAAA,GAAA,OAAA;AAEA,aAAA,UAAA,GAAA,KAAA;AACA,aAAA,OAAA,GAAA,IAAA;AAEA,aAAA,MAAA,GAAA,IAAA;AACA,aAAA,iBAAA,GAAA,IAAA;AAKN,aAAA,IAAA,IAAA,IAAA;AACD;AAED,KAAA,OAAA,IAAS;AACP,YAAI,EAAA,WAAA,KAAJ,IAAA;AAEA,YAAI,KAAA,UAAA,KAAJ,IAAA,EAA8B;AAC5B,gBAAI,cAAS,CAAC,aAAA,GAAA,CAAd,IAAc,CAAd,EAAuC;AACrC,sBAAM,IAAA,KAAA,CAAN,gCAAM,CAAN;AACD;AAED,iBAAA,WAAA,GAAmB,EAAnB,SAAA;AALF,SAAA,MAMO,IAAI,gBAAJ,SAAA,EAA+B;AACpC,iBAAA,UAAA,GAAA,IAAA;AACA,iBAAA,WAAA,GAAA,SAAA;AAEA,gBAAI;AACF,oBAAI,EAAA,OAAA,EAAA,MAAA,EAAA,iBAAA,EAAA,SAAA,EAAA,QAAA,KAAJ,IAAA;AAEA,oBAAI,WAAJ,IAAA,EAAqB;AACnB,wBAAI,cAAc,OAAlB,OAAkB,GAAlB;AAEA,wBAAI,gBAAJ,iBAAA,EAAuC;AACrC,mCAAW,KAAA,GAAA,CAAA,QAAA,EAAX,SAAW,CAAX;AADF,qBAAA,MAEO;AACL;AACA,6BAAA,iBAAA,GAAA,IAAA;AACA,mCAAW,KAAA,GAAA,CAAA,QAAA,EAAX,WAAW,CAAX;AACD;AACF;AAED,oBAAI,YAAJ,IAAA,EAAsB;AACpB,yBAAK,IAAI,IAAT,CAAA,EAAgB,IAAI,QAApB,MAAA,EAAA,GAAA,EAAyC;AACvC,4BAAI,QAAQ,QAAA,CAAA,EAAZ,OAAY,GAAZ;AACA,mCAAW,KAAA,GAAA,CAAA,KAAA,EAAX,QAAW,CAAX;AACD;AACF;AAED,qBAAA,SAAA,GAAA,QAAA;AAtBF,aAAA,SAuBU;AACR,qBAAA,UAAA,GAAA,KAAA;AACD;AACF;AAED,eAAO,KAAP,SAAA;AACD;AAED,WAAA,MAAA,CAAA,IAAA,EAAA,OAAA,EAA8C;AAC5C,YAAI,cAAS,KAAA,IAAA,MAAb,CAAA,CAAA,eAAA,EAA2D;AACzD,sBAAM,IAAA,KAAA,CAAN,kDAAM,CAAN;AACD;AAED;AACA,YAAI,MAAJ,IAAA;AACA,YAAI,SAAJ,OAAA;AAEA,YAAI,WAAJ,YAAA,EAA6B;AAC3B,gBAAA,MAAA,GAAA,IAAA;AADF,SAAA,MAEO;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAA,iBAAA,GAAwB,OAAxB,OAAwB,GAAxB;AACA,gBAAA,MAAA,GAAA,MAAA;AACD;AACF;AAED,WAAA,KAAA,CAAA,GAAA,EAA6C;AAC3C,YACE,cACA,EAAE,IAAA,IAAA,MAAA,CAAA,CAAA,eAAA,IAA+C,IAAA,IAAA,MAAjD,CAAA,CAFF,eAEE,CAFF,EAGE;AACA,kBAAM,IAAA,KAAA,CAAN,iDAAM,CAAN;AACD;AAED,YAAA,UAAA,EAAW;AACT;AACA;AACA,6CAAA,GAAA;AACD;AAEA,YAAA,QAAA,GAAsC,EAAtC,SAAA;AACF;AA/GqB;AAkHjB,MAAM,wBAAQ,mBAAd,KAAA;AACA,MAAM,0BAAS,mBAAf,MAAA;AAEP;AAEM,SAAA,SAAA,GAAmB;AACvB,WAAO,IAAA,kBAAA,CAAA,CAAA,CAAP,eAAO,CAAP;AACD;AAEK,SAAA,kBAAA,GAA4B;AAChC,WAAO,IAAA,kBAAA,CAAA,CAAA,CAAP,eAAO,CAAP;AACD;AAED;AAEO,MAAM,sCAAe,IAAA,kBAAA,CAAA,CAAA,CAArB,cAAqB,CAArB;AAED,SAAA,OAAA,CAAkB,EAAlB,GAAkB,EAAlB,EAAiC;AACrC,WAAO,QAAP,YAAA;AACD;AAEK,SAAA,UAAA,CAAA,GAAA,EAA6B;AACjC,WAAO,QAAP,YAAA;AACD;AAED;AAEA,MAAA,WAAA,CAAiB;AACf,KAAA,OAAA,IAAS;AACP,eAAA,QAAA;AACD;AAHc;AAMV,MAAM,sCAAe,IAArB,WAAqB,EAArB;AAEP;AAEA,MAAA,UAAA,CAAgB;AACd,KAAA,OAAA,IAAS;AACP,eAAA,SAAA;AACD;AAHa;AAMT,MAAM,oCAAc,IAApB,UAAoB,EAApB;AAEP;AAEM,SAAA,OAAA,CAAA,IAAA,EAA6B;AACjC,QAAI,YAAJ,EAAA;AAEA,SAAK,IAAI,IAAJ,CAAA,EAAW,IAAI,KAApB,MAAA,EAAiC,IAAjC,CAAA,EAAA,GAAA,EAA6C;AAC3C,YAAI,MAAM,KAAV,CAAU,CAAV;AACA,YAAI,QAAJ,YAAA,EAA0B;AAC1B,kBAAA,IAAA,CAAA,GAAA;AACD;AAED,WAAO,oBAAP,SAAO,CAAP;AACD;AAEK,SAAA,mBAAA,CAAA,IAAA,EAAyC;AAC7C,YAAQ,KAAR,MAAA;AACE,aAAA,CAAA;AACE,mBAAA,YAAA;AACF,aAAA,CAAA;AACE,mBAAO,KAAP,CAAO,CAAP;AACF;AACE,gBAAI,MAAM,IAAA,kBAAA,CAAA,CAAA,CAAV,gBAAU,CAAV;AACC,gBAAA,OAAA,GAAA,IAAA;AACD,mBAAA,GAAA;AARJ;AAUD","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 value(_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 validate(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 update(_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 dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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":""}

@@ -53,8 +53,3 @@ 'use strict';

*/
function value(tag) {
if (_env.DEBUG) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
function value(_tag) {
return $REVISION;

@@ -73,17 +68,4 @@ }

function validate(tag, snapshot) {
if (_env.DEBUG) {
IS_VALIDATING = true;
}
var isValid = snapshot >= tag[COMPUTE]();
if (_env.DEBUG) {
IS_VALIDATING = false;
if (isValid) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
}
return isValid;
return snapshot >= tag[COMPUTE]();
}
var IS_VALIDATING = void 0;
var TYPE = (0, _utils.symbol)('TAG_TYPE');

@@ -103,4 +85,5 @@ var ALLOW_CYCLES = exports.ALLOW_CYCLES = void 0;

this.isUpdating = false;
this.subtags = null;
this.subtag = null;
this.subtags = null;
this.subtagBufferCache = null;
this[TYPE] = type;

@@ -119,22 +102,19 @@ }

this.isUpdating = true;
if (_env.DEBUG) {
// In DEBUG, we don't cache while validating only, because it is valid
// update a tag between calling `validate()` and `value()`. Once you
// call `value()` on a tag, its revision is effectively locked in, and
// if you attempt to update it to a tag that is more recent it could
// break assumptions in our system. This is why the assertion exists in
// the static `update()` method below.
if (!IS_VALIDATING) {
this.lastChecked = $REVISION;
}
} else {
this.lastChecked = $REVISION;
}
this.lastChecked = $REVISION;
try {
var subtags = this.subtags,
subtag = this.subtag,
subtagBufferCache = this.subtagBufferCache,
lastValue = this.lastValue,
revision = this.revision;
if (subtag !== null) {
revision = Math.max(revision, subtag[COMPUTE]());
var subtagValue = subtag[COMPUTE]();
if (subtagValue === subtagBufferCache) {
revision = Math.max(revision, lastValue);
} else {
// Clear the temporary buffer cache
this.subtagBufferCache = null;
revision = Math.max(revision, subtagValue);
}
}

@@ -155,3 +135,3 @@ if (subtags !== null) {

MonomorphicTagImpl.update = function update(_tag, subtag) {
MonomorphicTagImpl.update = function update(_tag, _subtag) {
if (_env.DEBUG && _tag[TYPE] !== 1 /* Updatable */) {

@@ -162,8 +142,25 @@ throw new Error('Attempted to update a tag that was not updatable');

var tag = _tag;
var subtag = _subtag;
if (subtag === CONSTANT_TAG) {
tag.subtag = null;
} else {
if (_env.DEBUG && tag.lastChecked === $REVISION && subtag[COMPUTE]() > tag.lastValue) {
throw new Error('BUG: attempted to update a tag with a tag that has a more recent revision as its value');
}
// There are two different possibilities when updating a subtag:
//
// 1. subtag[COMPUTE]() <= tag[COMPUTE]();
// 2. subtag[COMPUTE]() > tag[COMPUTE]();
//
// The first possibility is completely fine within our caching model, but
// the second possibility presents a problem. If the parent tag has
// already been read, then it's value is cached and will not update to
// reflect the subtag's greater value. Next time the cache is busted, the
// subtag's value _will_ be read, and it's value will be _greater_ than
// the saved snapshot of the parent, causing the resulting calculation to
// be rerun erroneously.
//
// In order to prevent this, when we first update to a new subtag we store
// its computed value, and then check against that computed value on
// subsequent updates. If its value hasn't changed, then we return the
// parent's previous value. Once the subtag changes for the first time,
// we clear the cache and everything is finally in sync with the parent.
tag.subtagBufferCache = subtag[COMPUTE]();
tag.subtag = subtag;

@@ -259,2 +256,2 @@ }

}
//# 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 value(tag: Tag): Revision {\n  if (DEBUG) {\n    // compute to cache the latest value, which will prevent us from doing\n    // invalid updates later on.\n    tag[COMPUTE]();\n  }\n\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 validate(tag: Tag, snapshot: Revision) {\n  if (DEBUG) {\n    IS_VALIDATING = true;\n  }\n\n  let isValid = snapshot >= tag[COMPUTE]();\n\n  if (DEBUG) {\n    IS_VALIDATING = false;\n\n    if (isValid) {\n      // compute to cache the latest value, which will prevent us from doing\n      // invalid updates later on.\n      tag[COMPUTE]();\n    }\n  }\n\n  return isValid;\n}\n\nlet IS_VALIDATING: boolean | undefined;\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 subtag: Tag | null = null;\n  private subtags: Tag[] | 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\n      if (DEBUG) {\n        // In DEBUG, we don't cache while validating only, because it is valid\n        // update a tag between calling `validate()` and `value()`. Once you\n        // call `value()` on a tag, its revision is effectively locked in, and\n        // if you attempt to update it to a tag that is more recent it could\n        // break assumptions in our system. This is why the assertion exists in\n        // the static `update()` method below.\n        if (!IS_VALIDATING) {\n          this.lastChecked = $REVISION;\n        }\n      } else {\n        this.lastChecked = $REVISION;\n      }\n\n      try {\n        let { subtags, subtag, revision } = this;\n\n        if (subtag !== null) {\n          revision = Math.max(revision, subtag[COMPUTE]());\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 update(_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\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      if (\n        DEBUG &&\n        tag.lastChecked === $REVISION &&\n        (subtag as MonomorphicTagImpl)[COMPUTE]() > tag.lastValue\n      ) {\n        throw new Error(\n          'BUG: attempted to update a tag with a tag that has a more recent revision as its value'\n        );\n      }\n\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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":";;;;;;QAcM,I,GAAA,I;QAsCA,K,GAAA,K;QAcA,Q,GAAA,Q;QAqKA,S,GAAA,S;QAIA,kB,GAAA,kB;QAQA,O,GAAA,O;QAIA,U,GAAA,U;QA0BA,O,GAAA,O;QAYA,mB,GAAA,mB;;AA7RN;;AACA;;AACA;;;;;;;;AAMO,IAAM,8BAAN,CAAA;AACA,IAAM,4BAAN,CAAA;AACA,IAAM,8BAAN,gBAAA,C,CAA6C;AAEpD,IAAI,YAAJ,OAAA;AAEM,SAAA,IAAA,GAAc;AAClB;AACD;AAED;AAEO,IAAM,4BAAyB,mBAA/B,aAA+B,CAA/B;AAgBP;AAEA;;;;;;;;;;;;;;AAcM,SAAA,KAAA,CAAA,IAAA,EAAyB;AAC7B,WAAA,SAAA;AACD;AAED;;;;;;;;;;AAUM,SAAA,QAAA,CAAA,GAAA,EAAA,QAAA,EAA+C;AACnD,WAAO,YAAY,IAAnB,OAAmB,GAAnB;AACD;AAiBD,IAAM,OAAsB,mBAA5B,UAA4B,CAA5B;AAEO,IAAA,sCAAA,KAAA,CAAA;AAEP,IAAA,UAAA,EAAW;AACT,YAHK,YAGL,kBAAe,IAAf,OAAe,EAAf;AACD;;IAqBD,qB;AAaE,aAAA,kBAAA,CAAA,IAAA,EAAqC;AAAA,wBAAA,IAAA,EAAA,kBAAA;;AAZ7B,aAAA,QAAA,GAAA,OAAA;AACA,aAAA,WAAA,GAAA,OAAA;AACA,aAAA,SAAA,GAAA,OAAA;AAEA,aAAA,UAAA,GAAA,KAAA;AACA,aAAA,OAAA,GAAA,IAAA;AAEA,aAAA,MAAA,GAAA,IAAA;AACA,aAAA,iBAAA,GAAA,IAAA;AAKN,aAAA,IAAA,IAAA,IAAA;AACD;;iCAED,O,gBAAS;AAAA,YAAA,cAAA,KAAA,WAAA;;AAGP,YAAI,KAAA,UAAA,KAAJ,IAAA,EAA8B;AAC5B,gBAAI,cAAS,CAAC,aAAA,GAAA,CAAd,IAAc,CAAd,EAAuC;AACrC,sBAAM,IAAA,KAAA,CAAN,gCAAM,CAAN;AACD;AAED,iBAAA,WAAA,GAAmB,EAAnB,SAAA;AALF,SAAA,MAMO,IAAI,gBAAJ,SAAA,EAA+B;AACpC,iBAAA,UAAA,GAAA,IAAA;AACA,iBAAA,WAAA,GAAA,SAAA;AAEA,gBAAI;AAAA,oBAAA,UAAA,KAAA,OAAA;AAAA,oBAAA,SAAA,KAAA,MAAA;AAAA,oBAAA,oBAAA,KAAA,iBAAA;AAAA,oBAAA,YAAA,KAAA,SAAA;AAAA,oBAAA,WAAA,KAAA,QAAA;;AAGF,oBAAI,WAAJ,IAAA,EAAqB;AACnB,wBAAI,cAAc,OAAlB,OAAkB,GAAlB;AAEA,wBAAI,gBAAJ,iBAAA,EAAuC;AACrC,mCAAW,KAAA,GAAA,CAAA,QAAA,EAAX,SAAW,CAAX;AADF,qBAAA,MAEO;AACL;AACA,6BAAA,iBAAA,GAAA,IAAA;AACA,mCAAW,KAAA,GAAA,CAAA,QAAA,EAAX,WAAW,CAAX;AACD;AACF;AAED,oBAAI,YAAJ,IAAA,EAAsB;AACpB,yBAAK,IAAI,IAAT,CAAA,EAAgB,IAAI,QAApB,MAAA,EAAA,GAAA,EAAyC;AACvC,4BAAI,SAAQ,QAAA,CAAA,EAAZ,OAAY,GAAZ;AACA,mCAAW,KAAA,GAAA,CAAA,MAAA,EAAX,QAAW,CAAX;AACD;AACF;AAED,qBAAA,SAAA,GAAA,QAAA;AAtBF,aAAA,SAuBU;AACR,qBAAA,UAAA,GAAA,KAAA;AACD;AACF;AAED,eAAO,KAAP,SAAA;;;uBAGF,M,mBAAA,I,EAAA,O,EAA8C;AAC5C,YAAI,cAAS,KAAA,IAAA,MAAb,CAAA,CAAA,eAAA,EAA2D;AACzD,sBAAM,IAAA,KAAA,CAAN,kDAAM,CAAN;AACD;AAED;AACA,YAAI,MAAJ,IAAA;AACA,YAAI,SAAJ,OAAA;AAEA,YAAI,WAAJ,YAAA,EAA6B;AAC3B,gBAAA,MAAA,GAAA,IAAA;AADF,SAAA,MAEO;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAA,iBAAA,GAAwB,OAAxB,OAAwB,GAAxB;AACA,gBAAA,MAAA,GAAA,MAAA;AACD;;;uBAGH,K,kBAAA,G,EAA6C;AAC3C,YACE,cACA,EAAE,IAAA,IAAA,MAAA,CAAA,CAAA,eAAA,IAA+C,IAAA,IAAA,MAAjD,CAAA,CAFF,eAEE,CAFF,EAGE;AACA,kBAAM,IAAA,KAAA,CAAN,iDAAM,CAAN;AACD;AAED,YAAA,UAAA,EAAW;AACT;AACA;AACA,6CAAA,GAAA;AACD;AAEA,YAAA,QAAA,GAAsC,EAAtC,SAAA;;;;;;AAIE,IAAM,wBAAQ,mBAAd,KAAA;AACA,IAAM,0BAAS,mBAAf,MAAA;AAEP;AAEM,SAAA,SAAA,GAAmB;AACvB,WAAO,IAAA,kBAAA,CAAA,CAAA,CAAP,eAAO,CAAP;AACD;AAEK,SAAA,kBAAA,GAA4B;AAChC,WAAO,IAAA,kBAAA,CAAA,CAAA,CAAP,eAAO,CAAP;AACD;AAED;AAEO,IAAM,sCAAe,IAAA,kBAAA,CAAA,CAAA,CAArB,cAAqB,CAArB;AAED,SAAA,OAAA,CAAA,IAAA,EAAiC;AAAA,QAAjC,MAAiC,KAAjC,GAAiC;;AACrC,WAAO,QAAP,YAAA;AACD;AAEK,SAAA,UAAA,CAAA,GAAA,EAA6B;AACjC,WAAO,QAAP,YAAA;AACD;AAED;;IAEA,c;;;;;0BACE,O,gBAAS;AACP,eAAA,QAAA;;;;;;AAIG,IAAM,sCAAe,IAArB,WAAqB,EAArB;AAEP;;IAEA,a;;;;;yBACE,O,gBAAS;AACP,eAAA,SAAA;;;;;;AAIG,IAAM,oCAAc,IAApB,UAAoB,EAApB;AAEP;AAEM,SAAA,OAAA,CAAA,IAAA,EAA6B;AACjC,QAAI,YAAJ,EAAA;AAEA,SAAK,IAAI,IAAJ,CAAA,EAAW,IAAI,KAApB,MAAA,EAAiC,IAAjC,CAAA,EAAA,GAAA,EAA6C;AAC3C,YAAI,MAAM,KAAV,CAAU,CAAV;AACA,YAAI,QAAJ,YAAA,EAA0B;AAC1B,kBAAA,IAAA,CAAA,GAAA;AACD;AAED,WAAO,oBAAP,SAAO,CAAP;AACD;AAEK,SAAA,mBAAA,CAAA,IAAA,EAAyC;AAC7C,YAAQ,KAAR,MAAA;AACE,aAAA,CAAA;AACE,mBAAA,YAAA;AACF,aAAA,CAAA;AACE,mBAAO,KAAP,CAAO,CAAP;AACF;AACE,gBAAI,MAAM,IAAA,kBAAA,CAAA,CAAA,CAAV,gBAAU,CAAV;AACC,gBAAA,OAAA,GAAA,IAAA;AACD,mBAAA,GAAA;AARJ;AAUD","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 value(_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 validate(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 update(_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 dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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":""}

@@ -28,8 +28,3 @@ import { DEBUG } from '@glimmer/env';

*/
export function value(tag) {
if (DEBUG) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
export function value(_tag) {
return $REVISION;

@@ -48,17 +43,4 @@ }

export function validate(tag, snapshot) {
if (DEBUG) {
IS_VALIDATING = true;
}
let isValid = snapshot >= tag[COMPUTE]();
if (DEBUG) {
IS_VALIDATING = false;
if (isValid) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
}
return isValid;
return snapshot >= tag[COMPUTE]();
}
let IS_VALIDATING;
const TYPE = symbol('TAG_TYPE');

@@ -75,4 +57,5 @@ export let ALLOW_CYCLES;

this.isUpdating = false;
this.subtags = null;
this.subtag = null;
this.subtags = null;
this.subtagBufferCache = null;
this[TYPE] = type;

@@ -89,19 +72,14 @@ }

this.isUpdating = true;
if (DEBUG) {
// In DEBUG, we don't cache while validating only, because it is valid
// update a tag between calling `validate()` and `value()`. Once you
// call `value()` on a tag, its revision is effectively locked in, and
// if you attempt to update it to a tag that is more recent it could
// break assumptions in our system. This is why the assertion exists in
// the static `update()` method below.
if (!IS_VALIDATING) {
this.lastChecked = $REVISION;
}
} else {
this.lastChecked = $REVISION;
}
this.lastChecked = $REVISION;
try {
let { subtags, subtag, revision } = this;
let { subtags, subtag, subtagBufferCache, lastValue, revision } = this;
if (subtag !== null) {
revision = Math.max(revision, subtag[COMPUTE]());
let subtagValue = subtag[COMPUTE]();
if (subtagValue === subtagBufferCache) {
revision = Math.max(revision, lastValue);
} else {
// Clear the temporary buffer cache
this.subtagBufferCache = null;
revision = Math.max(revision, subtagValue);
}
}

@@ -121,3 +99,3 @@ if (subtags !== null) {

}
static update(_tag, subtag) {
static update(_tag, _subtag) {
if (DEBUG && _tag[TYPE] !== 1 /* Updatable */) {

@@ -128,8 +106,25 @@ throw new Error('Attempted to update a tag that was not updatable');

let tag = _tag;
let subtag = _subtag;
if (subtag === CONSTANT_TAG) {
tag.subtag = null;
} else {
if (DEBUG && tag.lastChecked === $REVISION && subtag[COMPUTE]() > tag.lastValue) {
throw new Error('BUG: attempted to update a tag with a tag that has a more recent revision as its value');
}
// There are two different possibilities when updating a subtag:
//
// 1. subtag[COMPUTE]() <= tag[COMPUTE]();
// 2. subtag[COMPUTE]() > tag[COMPUTE]();
//
// The first possibility is completely fine within our caching model, but
// the second possibility presents a problem. If the parent tag has
// already been read, then it's value is cached and will not update to
// reflect the subtag's greater value. Next time the cache is busted, the
// subtag's value _will_ be read, and it's value will be _greater_ than
// the saved snapshot of the parent, causing the resulting calculation to
// be rerun erroneously.
//
// In order to prevent this, when we first update to a new subtag we store
// its computed value, and then check against that computed value on
// subsequent updates. If its value hasn't changed, then we return the
// parent's previous value. Once the subtag changes for the first time,
// we clear the cache and everything is finally in sync with the parent.
tag.subtagBufferCache = subtag[COMPUTE]();
tag.subtag = subtag;

@@ -203,2 +198,2 @@ }

}
//# 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 value(tag: Tag): Revision {\n  if (DEBUG) {\n    // compute to cache the latest value, which will prevent us from doing\n    // invalid updates later on.\n    tag[COMPUTE]();\n  }\n\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 validate(tag: Tag, snapshot: Revision) {\n  if (DEBUG) {\n    IS_VALIDATING = true;\n  }\n\n  let isValid = snapshot >= tag[COMPUTE]();\n\n  if (DEBUG) {\n    IS_VALIDATING = false;\n\n    if (isValid) {\n      // compute to cache the latest value, which will prevent us from doing\n      // invalid updates later on.\n      tag[COMPUTE]();\n    }\n  }\n\n  return isValid;\n}\n\nlet IS_VALIDATING: boolean | undefined;\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 subtag: Tag | null = null;\n  private subtags: Tag[] | 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\n      if (DEBUG) {\n        // In DEBUG, we don't cache while validating only, because it is valid\n        // update a tag between calling `validate()` and `value()`. Once you\n        // call `value()` on a tag, its revision is effectively locked in, and\n        // if you attempt to update it to a tag that is more recent it could\n        // break assumptions in our system. This is why the assertion exists in\n        // the static `update()` method below.\n        if (!IS_VALIDATING) {\n          this.lastChecked = $REVISION;\n        }\n      } else {\n        this.lastChecked = $REVISION;\n      }\n\n      try {\n        let { subtags, subtag, revision } = this;\n\n        if (subtag !== null) {\n          revision = Math.max(revision, subtag[COMPUTE]());\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 update(_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\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      if (\n        DEBUG &&\n        tag.lastChecked === $REVISION &&\n        (subtag as MonomorphicTagImpl)[COMPUTE]() > tag.lastValue\n      ) {\n        throw new Error(\n          'BUG: attempted to update a tag with a tag that has a more recent revision as its value'\n        );\n      }\n\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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 { 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 value(_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 validate(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 update(_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 dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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":""}

@@ -30,8 +30,3 @@ function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }

*/
export function value(tag) {
if (DEBUG) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
export function value(_tag) {
return $REVISION;

@@ -50,17 +45,4 @@ }

export function validate(tag, snapshot) {
if (DEBUG) {
IS_VALIDATING = true;
}
var isValid = snapshot >= tag[COMPUTE]();
if (DEBUG) {
IS_VALIDATING = false;
if (isValid) {
// compute to cache the latest value, which will prevent us from doing
// invalid updates later on.
tag[COMPUTE]();
}
}
return isValid;
return snapshot >= tag[COMPUTE]();
}
var IS_VALIDATING = void 0;
var TYPE = symbol('TAG_TYPE');

@@ -80,4 +62,5 @@ export var ALLOW_CYCLES = void 0;

this.isUpdating = false;
this.subtags = null;
this.subtag = null;
this.subtags = null;
this.subtagBufferCache = null;
this[TYPE] = type;

@@ -96,22 +79,19 @@ }

this.isUpdating = true;
if (DEBUG) {
// In DEBUG, we don't cache while validating only, because it is valid
// update a tag between calling `validate()` and `value()`. Once you
// call `value()` on a tag, its revision is effectively locked in, and
// if you attempt to update it to a tag that is more recent it could
// break assumptions in our system. This is why the assertion exists in
// the static `update()` method below.
if (!IS_VALIDATING) {
this.lastChecked = $REVISION;
}
} else {
this.lastChecked = $REVISION;
}
this.lastChecked = $REVISION;
try {
var subtags = this.subtags,
subtag = this.subtag,
subtagBufferCache = this.subtagBufferCache,
lastValue = this.lastValue,
revision = this.revision;
if (subtag !== null) {
revision = Math.max(revision, subtag[COMPUTE]());
var subtagValue = subtag[COMPUTE]();
if (subtagValue === subtagBufferCache) {
revision = Math.max(revision, lastValue);
} else {
// Clear the temporary buffer cache
this.subtagBufferCache = null;
revision = Math.max(revision, subtagValue);
}
}

@@ -132,3 +112,3 @@ if (subtags !== null) {

MonomorphicTagImpl.update = function update(_tag, subtag) {
MonomorphicTagImpl.update = function update(_tag, _subtag) {
if (DEBUG && _tag[TYPE] !== 1 /* Updatable */) {

@@ -139,8 +119,25 @@ throw new Error('Attempted to update a tag that was not updatable');

var tag = _tag;
var subtag = _subtag;
if (subtag === CONSTANT_TAG) {
tag.subtag = null;
} else {
if (DEBUG && tag.lastChecked === $REVISION && subtag[COMPUTE]() > tag.lastValue) {
throw new Error('BUG: attempted to update a tag with a tag that has a more recent revision as its value');
}
// There are two different possibilities when updating a subtag:
//
// 1. subtag[COMPUTE]() <= tag[COMPUTE]();
// 2. subtag[COMPUTE]() > tag[COMPUTE]();
//
// The first possibility is completely fine within our caching model, but
// the second possibility presents a problem. If the parent tag has
// already been read, then it's value is cached and will not update to
// reflect the subtag's greater value. Next time the cache is busted, the
// subtag's value _will_ be read, and it's value will be _greater_ than
// the saved snapshot of the parent, causing the resulting calculation to
// be rerun erroneously.
//
// In order to prevent this, when we first update to a new subtag we store
// its computed value, and then check against that computed value on
// subsequent updates. If its value hasn't changed, then we return the
// parent's previous value. Once the subtag changes for the first time,
// we clear the cache and everything is finally in sync with the parent.
tag.subtagBufferCache = subtag[COMPUTE]();
tag.subtag = subtag;

@@ -236,2 +233,2 @@ }

}
//# 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 value(tag: Tag): Revision {\n  if (DEBUG) {\n    // compute to cache the latest value, which will prevent us from doing\n    // invalid updates later on.\n    tag[COMPUTE]();\n  }\n\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 validate(tag: Tag, snapshot: Revision) {\n  if (DEBUG) {\n    IS_VALIDATING = true;\n  }\n\n  let isValid = snapshot >= tag[COMPUTE]();\n\n  if (DEBUG) {\n    IS_VALIDATING = false;\n\n    if (isValid) {\n      // compute to cache the latest value, which will prevent us from doing\n      // invalid updates later on.\n      tag[COMPUTE]();\n    }\n  }\n\n  return isValid;\n}\n\nlet IS_VALIDATING: boolean | undefined;\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 subtag: Tag | null = null;\n  private subtags: Tag[] | 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\n      if (DEBUG) {\n        // In DEBUG, we don't cache while validating only, because it is valid\n        // update a tag between calling `validate()` and `value()`. Once you\n        // call `value()` on a tag, its revision is effectively locked in, and\n        // if you attempt to update it to a tag that is more recent it could\n        // break assumptions in our system. This is why the assertion exists in\n        // the static `update()` method below.\n        if (!IS_VALIDATING) {\n          this.lastChecked = $REVISION;\n        }\n      } else {\n        this.lastChecked = $REVISION;\n      }\n\n      try {\n        let { subtags, subtag, revision } = this;\n\n        if (subtag !== null) {\n          revision = Math.max(revision, subtag[COMPUTE]());\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 update(_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\n    if (subtag === CONSTANT_TAG) {\n      tag.subtag = null;\n    } else {\n      if (\n        DEBUG &&\n        tag.lastChecked === $REVISION &&\n        (subtag as MonomorphicTagImpl)[COMPUTE]() > tag.lastValue\n      ) {\n        throw new Error(\n          'BUG: attempted to update a tag with a tag that has a more recent revision as its value'\n        );\n      }\n\n      tag.subtag = subtag;\n    }\n  }\n\n  static dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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 { 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 value(_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 validate(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 update(_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 dirty(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 dirty = MonomorphicTagImpl.dirty;\nexport const update = MonomorphicTagImpl.update;\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 isConst({ 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\nclass VolatileTag implements Tag {\n  [COMPUTE]() {\n    return VOLATILE;\n  }\n}\n\nexport const VOLATILE_TAG = new VolatileTag();\n\n//////////\n\nclass 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":""}

@@ -33,3 +33,3 @@ import { UnionToIntersection } from './utils';

*/
export declare function value(tag: Tag): Revision;
export declare function value(_tag: Tag): Revision;
/**

@@ -84,8 +84,9 @@ * `validate` receives a tag and a snapshot from a previous call to `value` with

private isUpdating;
private subtags;
private subtag;
private subtags;
private subtagBufferCache;
[TYPE]: MonomorphicTagType;
constructor(type: MonomorphicTagTypes);
[COMPUTE](): Revision;
static update(_tag: UpdatableTag, subtag: Tag): void;
static update(_tag: UpdatableTag, _subtag: Tag): void;
static dirty(tag: DirtyableTag | UpdatableTag): void;

@@ -92,0 +93,0 @@ }

{
"name": "@glimmer/validator",
"version": "0.46.0",
"version": "0.47.0",
"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