@uniformdev/context 19.79.1-alpha.13 → 19.79.1-alpha.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/LICENSE.txt CHANGED
@@ -1,2 +1,2 @@
1
- © 2023 Uniform Systems, Inc. All Rights Reserved.
1
+ © 2024 Uniform Systems, Inc. All Rights Reserved.
2
2
  See details of Uniform Systems, Inc. Master Subscription Agreement here: https://uniform.dev/eula
@@ -1,4 +1,4 @@
1
- import { aC as paths$7, N as ManifestV2 } from '../types-8605X2cu.mjs';
1
+ import { aF as paths$7, N as ManifestV2 } from '../types-D4Fs5NVm.mjs';
2
2
  import 'mitt';
3
3
 
4
4
  type LimitPolicy = <ReturnValue>(func: () => Promise<ReturnValue>) => Promise<ReturnValue>;
package/dist/api/api.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- import { aC as paths$7, N as ManifestV2 } from '../types-8605X2cu.js';
1
+ import { aF as paths$7, N as ManifestV2 } from '../types-D4Fs5NVm.js';
2
2
  import 'mitt';
3
3
 
4
4
  type LimitPolicy = <ReturnValue>(func: () => Promise<ReturnValue>) => Promise<ReturnValue>;
package/dist/index.d.mts CHANGED
@@ -1,5 +1,5 @@
1
- import { O as OutputSeverity, L as LogDrain, C as ContextPlugin, S as ScoreVector, A as AggregateDimension, T as TransitionDataStore, a as StorageCommands, V as VisitorData, b as TransitionDataStoreOptions, D as DecayFunction, c as CriteriaEvaluator, d as StringMatch, e as VariantMatchCriteria, f as LogMessage, g as DevToolsEvents } from './types-8605X2cu.mjs';
2
- export { _ as AggregateDimensionInput, a8 as BehaviorTag, h as CONTEXTUAL_EDITING_TEST_NAME, i as CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID, n as Context, l as ContextEvents, m as ContextInstance, j as ContextOptions, az as ContextState, aA as ContextStateUpdate, J as CriteriaEvaluatorParameters, I as CriteriaEvaluatorResult, ap as DecayOptions, q as DevToolsActions, t as DevToolsDataEvent, r as DevToolsEvent, x as DevToolsForgetEvent, u as DevToolsHelloEvent, s as DevToolsLogEvent, w as DevToolsRawCommandsEvent, p as DevToolsState, o as DevToolsUiVersion, v as DevToolsUpdateEvent, a5 as DimensionMatch, X as EnrichmentCategory, aw as EnrichmentData, a1 as EnrichmentGoal, ax as EventData, $ as Goal, aB as GoalStateUpdate, av as Goals, H as GroupCriteriaEvaluator, ak as IdentifyCommand, F as LogMessageGroup, E as LogMessageSingle, y as LogMessages, G as ManifestInstance, N as ManifestV2, M as MessageCategory, B as MessageFunc, af as ModifyScoreCommand, ag as ModifySessionScoreCommand, Y as NumberMatch, P as PersonalizationEvent, Q as PersonalizationManifest, a6 as PersonalizeOptions, aa as PersonalizedResult, a9 as PersonalizedVariant, a2 as QuirkGoal, at as Quirks, an as SERVER_STATE_ID, am as ServerToClientTransitionState, ah as SetConsentCommand, al as SetControlGroupCommand, ae as SetGoalCommand, ai as SetQuirkCommand, aj as SetTestCommand, z as Severity, R as Signal, W as SignalCriteria, U as SignalCriteriaGroup, K as SignalData, a0 as SignalGoal, ad as StorageCommand, Z as TestDefinition, k as TestEvent, a3 as TestOptions, ac as TestResult, ab as TestVariant, au as Tests, ao as TransitionDataStoreEvents, as as VisitorDataStore, ar as VisitorDataStoreEvents, aq as VisitorDataStoreOptions, ay as emptyVisitorData, a7 as personalizeVariations, a4 as testVariations } from './types-8605X2cu.mjs';
1
+ import { O as OutputSeverity, L as LogDrain, C as ContextPlugin, S as ScoreVector, A as AggregateDimension, T as TransitionDataStore, a as StorageCommands, V as VisitorData, b as TransitionDataStoreOptions, D as DecayFunction, c as CriteriaEvaluator, d as StringMatch, e as VariantMatchCriteria, f as LogMessage, g as DevToolsEvents } from './types-D4Fs5NVm.mjs';
2
+ export { _ as AggregateDimensionInput, a8 as BehaviorTag, h as CONTEXTUAL_EDITING_TEST_NAME, i as CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID, n as Context, l as ContextEvents, m as ContextInstance, j as ContextOptions, aC as ContextState, aD as ContextStateUpdate, J as CriteriaEvaluatorParameters, I as CriteriaEvaluatorResult, aq as DecayOptions, q as DevToolsActions, t as DevToolsDataEvent, r as DevToolsEvent, x as DevToolsForgetEvent, u as DevToolsHelloEvent, s as DevToolsLogEvent, w as DevToolsRawCommandsEvent, p as DevToolsState, o as DevToolsUiVersion, v as DevToolsUpdateEvent, a5 as DimensionMatch, X as EnrichmentCategory, ax as EnrichmentData, a1 as EnrichmentGoal, aA as EventData, $ as Goal, aE as GoalStateUpdate, aw as Goals, H as GroupCriteriaEvaluator, ak as IdentifyCommand, F as LogMessageGroup, E as LogMessageSingle, y as LogMessages, G as ManifestInstance, N as ManifestV2, M as MessageCategory, B as MessageFunc, af as ModifyScoreCommand, ag as ModifySessionScoreCommand, Y as NumberMatch, P as PersonalizationEvent, Q as PersonalizationManifest, ay as PersonalizeControlVariant, a6 as PersonalizeOptions, az as PersonalizeVariants, aa as PersonalizedResult, a9 as PersonalizedVariant, a2 as QuirkGoal, au as Quirks, ao as SERVER_STATE_ID, an as ServerToClientTransitionState, ah as SetConsentCommand, al as SetControlGroupCommand, ae as SetGoalCommand, am as SetPersonalizeVariantControlCommand, ai as SetQuirkCommand, aj as SetTestCommand, z as Severity, R as Signal, W as SignalCriteria, U as SignalCriteriaGroup, K as SignalData, a0 as SignalGoal, ad as StorageCommand, Z as TestDefinition, k as TestEvent, a3 as TestOptions, ac as TestResult, ab as TestVariant, av as Tests, ap as TransitionDataStoreEvents, at as VisitorDataStore, as as VisitorDataStoreEvents, ar as VisitorDataStoreOptions, aB as emptyVisitorData, a7 as personalizeVariations, a4 as testVariations } from './types-D4Fs5NVm.mjs';
3
3
  import Cookies from 'js-cookie';
4
4
  import 'mitt';
5
5
 
package/dist/index.d.ts CHANGED
@@ -1,5 +1,5 @@
1
- import { O as OutputSeverity, L as LogDrain, C as ContextPlugin, S as ScoreVector, A as AggregateDimension, T as TransitionDataStore, a as StorageCommands, V as VisitorData, b as TransitionDataStoreOptions, D as DecayFunction, c as CriteriaEvaluator, d as StringMatch, e as VariantMatchCriteria, f as LogMessage, g as DevToolsEvents } from './types-8605X2cu.js';
2
- export { _ as AggregateDimensionInput, a8 as BehaviorTag, h as CONTEXTUAL_EDITING_TEST_NAME, i as CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID, n as Context, l as ContextEvents, m as ContextInstance, j as ContextOptions, az as ContextState, aA as ContextStateUpdate, J as CriteriaEvaluatorParameters, I as CriteriaEvaluatorResult, ap as DecayOptions, q as DevToolsActions, t as DevToolsDataEvent, r as DevToolsEvent, x as DevToolsForgetEvent, u as DevToolsHelloEvent, s as DevToolsLogEvent, w as DevToolsRawCommandsEvent, p as DevToolsState, o as DevToolsUiVersion, v as DevToolsUpdateEvent, a5 as DimensionMatch, X as EnrichmentCategory, aw as EnrichmentData, a1 as EnrichmentGoal, ax as EventData, $ as Goal, aB as GoalStateUpdate, av as Goals, H as GroupCriteriaEvaluator, ak as IdentifyCommand, F as LogMessageGroup, E as LogMessageSingle, y as LogMessages, G as ManifestInstance, N as ManifestV2, M as MessageCategory, B as MessageFunc, af as ModifyScoreCommand, ag as ModifySessionScoreCommand, Y as NumberMatch, P as PersonalizationEvent, Q as PersonalizationManifest, a6 as PersonalizeOptions, aa as PersonalizedResult, a9 as PersonalizedVariant, a2 as QuirkGoal, at as Quirks, an as SERVER_STATE_ID, am as ServerToClientTransitionState, ah as SetConsentCommand, al as SetControlGroupCommand, ae as SetGoalCommand, ai as SetQuirkCommand, aj as SetTestCommand, z as Severity, R as Signal, W as SignalCriteria, U as SignalCriteriaGroup, K as SignalData, a0 as SignalGoal, ad as StorageCommand, Z as TestDefinition, k as TestEvent, a3 as TestOptions, ac as TestResult, ab as TestVariant, au as Tests, ao as TransitionDataStoreEvents, as as VisitorDataStore, ar as VisitorDataStoreEvents, aq as VisitorDataStoreOptions, ay as emptyVisitorData, a7 as personalizeVariations, a4 as testVariations } from './types-8605X2cu.js';
1
+ import { O as OutputSeverity, L as LogDrain, C as ContextPlugin, S as ScoreVector, A as AggregateDimension, T as TransitionDataStore, a as StorageCommands, V as VisitorData, b as TransitionDataStoreOptions, D as DecayFunction, c as CriteriaEvaluator, d as StringMatch, e as VariantMatchCriteria, f as LogMessage, g as DevToolsEvents } from './types-D4Fs5NVm.js';
2
+ export { _ as AggregateDimensionInput, a8 as BehaviorTag, h as CONTEXTUAL_EDITING_TEST_NAME, i as CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID, n as Context, l as ContextEvents, m as ContextInstance, j as ContextOptions, aC as ContextState, aD as ContextStateUpdate, J as CriteriaEvaluatorParameters, I as CriteriaEvaluatorResult, aq as DecayOptions, q as DevToolsActions, t as DevToolsDataEvent, r as DevToolsEvent, x as DevToolsForgetEvent, u as DevToolsHelloEvent, s as DevToolsLogEvent, w as DevToolsRawCommandsEvent, p as DevToolsState, o as DevToolsUiVersion, v as DevToolsUpdateEvent, a5 as DimensionMatch, X as EnrichmentCategory, ax as EnrichmentData, a1 as EnrichmentGoal, aA as EventData, $ as Goal, aE as GoalStateUpdate, aw as Goals, H as GroupCriteriaEvaluator, ak as IdentifyCommand, F as LogMessageGroup, E as LogMessageSingle, y as LogMessages, G as ManifestInstance, N as ManifestV2, M as MessageCategory, B as MessageFunc, af as ModifyScoreCommand, ag as ModifySessionScoreCommand, Y as NumberMatch, P as PersonalizationEvent, Q as PersonalizationManifest, ay as PersonalizeControlVariant, a6 as PersonalizeOptions, az as PersonalizeVariants, aa as PersonalizedResult, a9 as PersonalizedVariant, a2 as QuirkGoal, au as Quirks, ao as SERVER_STATE_ID, an as ServerToClientTransitionState, ah as SetConsentCommand, al as SetControlGroupCommand, ae as SetGoalCommand, am as SetPersonalizeVariantControlCommand, ai as SetQuirkCommand, aj as SetTestCommand, z as Severity, R as Signal, W as SignalCriteria, U as SignalCriteriaGroup, K as SignalData, a0 as SignalGoal, ad as StorageCommand, Z as TestDefinition, k as TestEvent, a3 as TestOptions, ac as TestResult, ab as TestVariant, av as Tests, ap as TransitionDataStoreEvents, at as VisitorDataStore, as as VisitorDataStoreEvents, ar as VisitorDataStoreOptions, aB as emptyVisitorData, a7 as personalizeVariations, a4 as testVariations } from './types-D4Fs5NVm.js';
3
3
  import Cookies from 'js-cookie';
4
4
  import 'mitt';
5
5
 
package/dist/index.esm.js CHANGED
@@ -522,6 +522,15 @@ var SignalInstance = class {
522
522
  _evaluator = new WeakMap();
523
523
  _onLogMessage = new WeakMap();
524
524
 
525
+ // src/manifest/utils/control.ts
526
+ var rollForControlGroup = (value) => {
527
+ let control = value;
528
+ if (control >= 1) {
529
+ control = control / 100;
530
+ }
531
+ return Math.random() < control;
532
+ };
533
+
525
534
  // src/manifest/ManifestInstance.ts
526
535
  var _mf, _signalInstances, _goalEvaluators, _onLogMessage2;
527
536
  var ManifestInstance = class {
@@ -556,12 +565,8 @@ var ManifestInstance = class {
556
565
  __privateSet(this, _onLogMessage2, onLogMessage);
557
566
  }
558
567
  rollForControlGroup() {
559
- var _a, _b;
560
- let control = (_b = (_a = __privateGet(this, _mf).pz) == null ? void 0 : _a.control) != null ? _b : 0;
561
- if (control >= 1) {
562
- control = control / 100;
563
- }
564
- return Math.random() < control;
568
+ var _a;
569
+ return rollForControlGroup(((_a = __privateGet(this, _mf).pz) == null ? void 0 : _a.control) || 0);
565
570
  }
566
571
  getTest(name) {
567
572
  var _a;
@@ -643,7 +648,30 @@ function evaluateDimensionMatch(crit, vec, onLogMessage) {
643
648
  var _a, _b;
644
649
  const { op, l: lhs } = crit;
645
650
  const lhsScore = (_a = vec[lhs]) != null ? _a : 0;
646
- if (op === "+") {
651
+ if (op === "^") {
652
+ const [cat] = lhs.split(ENR_SEPARATOR);
653
+ let topVectorName = void 0;
654
+ let topScore = 0;
655
+ Object.keys(vec).forEach((vectorName) => {
656
+ if (vectorName.startsWith(`${cat}${ENR_SEPARATOR}`)) {
657
+ const score = vec[vectorName];
658
+ if (score > topScore) {
659
+ topVectorName = vectorName;
660
+ topScore = score;
661
+ }
662
+ }
663
+ });
664
+ const result = topVectorName === lhs;
665
+ onLogMessage == null ? void 0 : onLogMessage([
666
+ "info",
667
+ 302,
668
+ {
669
+ matched: result,
670
+ description: `${crit.l} has the highest score in the category`
671
+ }
672
+ ]);
673
+ return result;
674
+ } else if (op === "+") {
647
675
  const result = Math.max(...Object.values(vec)) === lhsScore && lhsScore > 0;
648
676
  onLogMessage == null ? void 0 : onLogMessage([
649
677
  "info",
@@ -730,31 +758,70 @@ function personalizeVariations({
730
758
  take = 1,
731
759
  onLogMessage
732
760
  }) {
733
- var _a, _b, _c;
761
+ var _a, _b, _c, _d;
734
762
  onLogMessage == null ? void 0 : onLogMessage(["info", 300, "GROUP", { name, take }]);
735
763
  try {
736
764
  const control = (_a = context.storage.data.controlGroup) != null ? _a : false;
737
765
  const results = [];
738
766
  let personalized = false;
739
767
  const scores = context.scores;
768
+ let index = 0;
769
+ const defaultVariants = [];
770
+ for (const variant of variations) {
771
+ if (!((_b = variant.pz) == null ? void 0 : _b.crit.length)) {
772
+ defaultVariants.push(variant);
773
+ }
774
+ }
740
775
  for (const variant of variations) {
776
+ const currentIndex = index++;
741
777
  if (results.length === take) {
742
778
  break;
743
779
  }
744
- if (!((_b = variant.pz) == null ? void 0 : _b.crit.length)) {
745
- onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, op: (_c = variant.pz) == null ? void 0 : _c.op }]);
780
+ if (!((_c = variant.pz) == null ? void 0 : _c.crit.length)) {
781
+ onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, op: (_d = variant.pz) == null ? void 0 : _d.op }]);
746
782
  onLogMessage == null ? void 0 : onLogMessage(["info", 302, { matched: true, description: "default variation" }]);
747
783
  onLogMessage == null ? void 0 : onLogMessage(["info", 303, true]);
748
784
  onLogMessage == null ? void 0 : onLogMessage(["info", 301, "ENDGROUP"]);
749
- results.push(variant);
785
+ results.push({
786
+ ...variant,
787
+ control: false
788
+ });
750
789
  continue;
751
790
  }
752
791
  if (control) {
753
792
  continue;
754
793
  }
755
794
  if (evaluateVariantMatch(variant.id, variant.pz, scores, onLogMessage)) {
756
- personalized = true;
757
- results.push(variant);
795
+ let variantToAdd = variant;
796
+ let isControl = false;
797
+ const isDefault = defaultVariants.find((v) => v.id === variant.id);
798
+ if (take === 1 && !isDefault && defaultVariants.length && typeof variant.pz.control === "number") {
799
+ isControl = context.getPersonalizeVariantControl(name, currentIndex);
800
+ if (typeof isControl === "undefined") {
801
+ isControl = rollForControlGroup(variant.pz.control);
802
+ context.storage.updateData([
803
+ {
804
+ type: "setpersonalizecontrol",
805
+ data: {
806
+ personlizationName: name,
807
+ index: currentIndex,
808
+ control: isControl
809
+ }
810
+ }
811
+ ]);
812
+ }
813
+ if (isControl) {
814
+ variantToAdd = {
815
+ ...defaultVariants[0],
816
+ id: variant.id
817
+ };
818
+ }
819
+ }
820
+ personalized = personalized || typeof variantToAdd.pz !== "undefined";
821
+ results.push({
822
+ ...variantToAdd,
823
+ control: isControl
824
+ });
758
825
  }
759
826
  }
760
827
  return {
@@ -1001,16 +1068,19 @@ function parseScoreCookie(cookieValue) {
1001
1068
  if (!cookieValue)
1002
1069
  return;
1003
1070
  const types = cookieValue.split(TYPE_SEP);
1004
- if (types.length > 3)
1071
+ if (types.length > 5)
1005
1072
  return;
1006
- const [abTestData, sessionScores, visitorScores] = types;
1007
- return {
1073
+ const [abTestData, sessionScores, visitorScores, controlGroup, personalizeVariants] = types;
1074
+ const data = {
1008
1075
  // this is true since we're reading a cookie, which wouldn't exist if consent wasn't given
1009
1076
  consent: true,
1010
1077
  sessionScores: decodeCookieType(parseCookieType(sessionScores)),
1011
1078
  scores: decodeCookieType(parseCookieType(visitorScores)),
1012
- tests: parseCookieType(abTestData)
1079
+ tests: parseCookieType(abTestData),
1080
+ controlGroup: controlGroup === "1",
1081
+ personalizeVariants: decodePersonalizeVariants(personalizeVariants)
1013
1082
  };
1083
+ return data;
1014
1084
  }
1015
1085
  function parseCookieType(type) {
1016
1086
  if (!type) {
@@ -1034,9 +1104,48 @@ function serializeCookie(data) {
1034
1104
  return [
1035
1105
  serializeCookieType(data.tests),
1036
1106
  serializeCookieType(encodeCookieType(data.sessionScores)),
1037
- serializeCookieType(encodeCookieType(data.scores))
1107
+ serializeCookieType(encodeCookieType(data.scores)),
1108
+ data.controlGroup ? "1" : "0",
1109
+ serializePersonalizeVariants(data)
1038
1110
  ].join(TYPE_SEP);
1039
1111
  }
1112
+ function serializePersonalizeVariants({
1113
+ personalizeVariants
1114
+ }) {
1115
+ const data = {};
1116
+ if (typeof personalizeVariants === "object") {
1117
+ Object.keys(personalizeVariants).forEach((personalizationName) => {
1118
+ const results = [];
1119
+ const variants = personalizeVariants[personalizationName];
1120
+ variants.forEach((variant) => {
1121
+ results.push(`${variant.index}:${variant.control ? "1" : "0"}`);
1122
+ });
1123
+ data[personalizationName] = results.join(",");
1124
+ });
1125
+ }
1126
+ const serialized = serializeCookieType(data);
1127
+ return serialized;
1128
+ }
1129
+ function decodePersonalizeVariants(data) {
1130
+ const parsed = parseCookieType(data);
1131
+ const keys = Object.keys(parsed);
1132
+ if (!keys.length) {
1133
+ return void 0;
1134
+ }
1135
+ const results = {};
1136
+ Object.keys(parsed).forEach((k) => {
1137
+ const variants = parsed[k].split(",");
1138
+ const key = decodeURIComponent(k);
1139
+ results[key] = variants.map((variant) => {
1140
+ const [index, control] = variant.split(":");
1141
+ return {
1142
+ index: parseInt(index, 10),
1143
+ control: control === "1"
1144
+ };
1145
+ });
1146
+ });
1147
+ return results;
1148
+ }
1040
1149
  function encodeCookieType(type) {
1041
1150
  return Object.entries(type).reduce((acc, [key, value]) => {
1042
1151
  acc[key] = ntob(value);
@@ -1125,7 +1234,8 @@ var emptyVisitorData = () => ({
1125
1234
  tests: {},
1126
1235
  goals: {},
1127
1236
  consent: false,
1128
- controlGroup: false
1237
+ controlGroup: false,
1238
+ personalizeVariants: {}
1129
1239
  });
1130
1240
 
1131
1241
  // src/storage/VisitorDataStore.ts
@@ -1174,6 +1284,25 @@ function applyCommandsToData(commands, state, inControlGroup) {
1174
1284
  case "setcontrol":
1175
1285
  newData.controlGroup = command.data;
1176
1286
  break;
1287
+ case "setpersonalizecontrol":
1288
+ if (!newData.personalizeVariants) {
1289
+ newData.personalizeVariants = {};
1290
+ }
1291
+ if (!newData.personalizeVariants[command.data.personlizationName]) {
1292
+ newData.personalizeVariants[command.data.personlizationName] = [];
1293
+ }
1294
+ const existingDef = newData.personalizeVariants[command.data.personlizationName].find(
1295
+ (i) => i.index === command.data.index
1296
+ );
1297
+ if (!existingDef) {
1298
+ newData.personalizeVariants[command.data.personlizationName].push({
1299
+ index: command.data.index,
1300
+ control: command.data.control
1301
+ });
1302
+ } else {
1303
+ console.warn("Overwriting existing control group definition is not allowed");
1304
+ }
1305
+ break;
1177
1306
  case "setgoal":
1178
1307
  newData.goals = (_c = newData.goals) != null ? _c : {};
1179
1308
  newData.goals[command.data.goal] = true;
@@ -1556,7 +1685,7 @@ var Context = class {
1556
1685
  * will NOT result in a recomputation of signal state.
1557
1686
  */
1558
1687
  async update(newData) {
1559
- var _a, _b, _c;
1688
+ var _a, _b, _c, _d;
1560
1689
  const commands = [];
1561
1690
  const newServerSideTests = {};
1562
1691
  if ((_a = __privateGet(this, _serverTransitionState)) == null ? void 0 : _a.quirks) {
@@ -1586,6 +1715,17 @@ var Context = class {
1586
1715
  );
1587
1716
  }
1588
1717
  }
1718
+ if ((_c = __privateGet(this, _serverTransitionState)) == null ? void 0 : _c.personalizeVariants) {
1719
+ Object.keys(__privateGet(this, _serverTransitionState).personalizeVariants).forEach((personalizationName) => {
1720
+ const variants = __privateGet(this, _serverTransitionState).personalizeVariants[personalizationName];
1721
+ variants.forEach((e) => {
1722
+ commands.push({
1723
+ type: "setpersonalizecontrol",
1724
+ data: { personlizationName: personalizationName, index: e.index, control: e.control }
1725
+ });
1726
+ });
1727
+ });
1728
+ }
1589
1729
  try {
1590
1730
  __privateGet(this, _mitt3).emit("log", [
1591
1731
  "info",
@@ -1595,7 +1735,7 @@ var Context = class {
1595
1735
  ...newData,
1596
1736
  // need to convert url to string so it can be json serialized
1597
1737
  // to go over postMessage to chrome extension
1598
- url: (_c = newData.url) == null ? void 0 : _c.toString()
1738
+ url: (_d = newData.url) == null ? void 0 : _d.toString()
1599
1739
  }
1600
1740
  ]);
1601
1741
  if (newData.quirks) {
@@ -1686,6 +1826,13 @@ var Context = class {
1686
1826
  }
1687
1827
  ]);
1688
1828
  }
1829
+ getPersonalizeVariantControl(name, index) {
1830
+ var _a, _b, _c;
1831
+ const source = (_b = (_a = __privateGet(this, _serverTransitionState)) == null ? void 0 : _a.personalizeVariants) != null ? _b : this.storage.data.personalizeVariants;
1832
+ const variants = (_c = source == null ? void 0 : source[name]) != null ? _c : [];
1833
+ const variant = variants.find((v) => v.index === index);
1834
+ return variant == null ? void 0 : variant.control;
1835
+ }
1689
1836
  /**
1690
1837
  * Writes a message to the Context log sink.
1691
1838
  * Used by Uniform internal SDK; not intended for public use.
@@ -1728,10 +1875,10 @@ var Context = class {
1728
1875
  const previousPlacement = __privateGet(this, _pzCache)[options.name];
1729
1876
  const eventData = {
1730
1877
  name: options.name,
1731
- variantIds: value.variations.map((variation) => {
1732
- var _a;
1733
- return (_a = variation.id) != null ? _a : "Unknown";
1734
- }),
1878
+ variantIds: value.variations.map((variation) => ({
1879
+ id: variation.id || "Unknown",
1880
+ control: variation.control
1881
+ })),
1735
1882
  control: this.storage.data.controlGroup,
1736
1883
  changed: true
1737
1884
  };
@@ -1768,7 +1915,8 @@ var Context = class {
1768
1915
  const transitionState = {
1769
1916
  quirks: this.storage.data.quirks,
1770
1917
  ssv: __privateGet(this, _scores),
1771
- tests: {}
1918
+ tests: {},
1919
+ personalizeVariants: this.storage.data.personalizeVariants
1772
1920
  };
1773
1921
  const allTests = this.storage.data.tests;
1774
1922
  Object.entries(allTests).map(([testName, testValue]) => {
@@ -2468,7 +2616,7 @@ var createInsightsClient = ({ endpoint }) => {
2468
2616
  return sendMessage(message);
2469
2617
  },
2470
2618
  personalizationResult: async (options) => {
2471
- const messages = options.variantIds.map((variantId) => {
2619
+ const messages = options.variantIds.map((variant) => {
2472
2620
  const message = {
2473
2621
  action: "personalization_result",
2474
2622
  version: "1",
@@ -2479,8 +2627,8 @@ var createInsightsClient = ({ endpoint }) => {
2479
2627
  payload: {
2480
2628
  ...getBasePayload(),
2481
2629
  name: options.name,
2482
- variantId,
2483
- control: options.control,
2630
+ variantId: variant.id,
2631
+ control: variant.control || options.control,
2484
2632
  changed: options.changed
2485
2633
  }
2486
2634
  };
package/dist/index.js CHANGED
@@ -593,6 +593,15 @@ var SignalInstance = class {
593
593
  _evaluator = new WeakMap();
594
594
  _onLogMessage = new WeakMap();
595
595
 
596
+ // src/manifest/utils/control.ts
597
+ var rollForControlGroup = (value) => {
598
+ let control = value;
599
+ if (control >= 1) {
600
+ control = control / 100;
601
+ }
602
+ return Math.random() < control;
603
+ };
604
+
596
605
  // src/manifest/ManifestInstance.ts
597
606
  var _mf, _signalInstances, _goalEvaluators, _onLogMessage2;
598
607
  var ManifestInstance = class {
@@ -627,12 +636,8 @@ var ManifestInstance = class {
627
636
  __privateSet(this, _onLogMessage2, onLogMessage);
628
637
  }
629
638
  rollForControlGroup() {
630
- var _a, _b;
631
- let control = (_b = (_a = __privateGet(this, _mf).pz) == null ? void 0 : _a.control) != null ? _b : 0;
632
- if (control >= 1) {
633
- control = control / 100;
634
- }
635
- return Math.random() < control;
639
+ var _a;
640
+ return rollForControlGroup(((_a = __privateGet(this, _mf).pz) == null ? void 0 : _a.control) || 0);
636
641
  }
637
642
  getTest(name) {
638
643
  var _a;
@@ -714,7 +719,30 @@ function evaluateDimensionMatch(crit, vec, onLogMessage) {
714
719
  var _a, _b;
715
720
  const { op, l: lhs } = crit;
716
721
  const lhsScore = (_a = vec[lhs]) != null ? _a : 0;
717
- if (op === "+") {
722
+ if (op === "^") {
723
+ const [cat] = lhs.split(ENR_SEPARATOR);
724
+ let topVectorName = void 0;
725
+ let topScore = 0;
726
+ Object.keys(vec).forEach((vectorName) => {
727
+ if (vectorName.startsWith(`${cat}${ENR_SEPARATOR}`)) {
728
+ const score = vec[vectorName];
729
+ if (score > topScore) {
730
+ topVectorName = vectorName;
731
+ topScore = score;
732
+ }
733
+ }
734
+ });
735
+ const result = topVectorName === lhs;
736
+ onLogMessage == null ? void 0 : onLogMessage([
737
+ "info",
738
+ 302,
739
+ {
740
+ matched: result,
741
+ description: `${crit.l} has the highest score in the category`
742
+ }
743
+ ]);
744
+ return result;
745
+ } else if (op === "+") {
718
746
  const result = Math.max(...Object.values(vec)) === lhsScore && lhsScore > 0;
719
747
  onLogMessage == null ? void 0 : onLogMessage([
720
748
  "info",
@@ -801,31 +829,70 @@ function personalizeVariations({
801
829
  take = 1,
802
830
  onLogMessage
803
831
  }) {
804
- var _a, _b, _c;
832
+ var _a, _b, _c, _d;
805
833
  onLogMessage == null ? void 0 : onLogMessage(["info", 300, "GROUP", { name, take }]);
806
834
  try {
807
835
  const control = (_a = context.storage.data.controlGroup) != null ? _a : false;
808
836
  const results = [];
809
837
  let personalized = false;
810
838
  const scores = context.scores;
839
+ let index = 0;
840
+ const defaultVariants = [];
841
+ for (const variant of variations) {
842
+ if (!((_b = variant.pz) == null ? void 0 : _b.crit.length)) {
843
+ defaultVariants.push(variant);
844
+ }
845
+ }
811
846
  for (const variant of variations) {
847
+ const currentIndex = index++;
812
848
  if (results.length === take) {
813
849
  break;
814
850
  }
815
- if (!((_b = variant.pz) == null ? void 0 : _b.crit.length)) {
816
- onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, op: (_c = variant.pz) == null ? void 0 : _c.op }]);
851
+ if (!((_c = variant.pz) == null ? void 0 : _c.crit.length)) {
852
+ onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, op: (_d = variant.pz) == null ? void 0 : _d.op }]);
817
853
  onLogMessage == null ? void 0 : onLogMessage(["info", 302, { matched: true, description: "default variation" }]);
818
854
  onLogMessage == null ? void 0 : onLogMessage(["info", 303, true]);
819
855
  onLogMessage == null ? void 0 : onLogMessage(["info", 301, "ENDGROUP"]);
820
- results.push(variant);
856
+ results.push({
857
+ ...variant,
858
+ control: false
859
+ });
821
860
  continue;
822
861
  }
823
862
  if (control) {
824
863
  continue;
825
864
  }
826
865
  if (evaluateVariantMatch(variant.id, variant.pz, scores, onLogMessage)) {
827
- personalized = true;
828
- results.push(variant);
866
+ let variantToAdd = variant;
867
+ let isControl = false;
868
+ const isDefault = defaultVariants.find((v) => v.id === variant.id);
869
+ if (take === 1 && !isDefault && defaultVariants.length && typeof variant.pz.control === "number") {
870
+ isControl = context.getPersonalizeVariantControl(name, currentIndex);
871
+ if (typeof isControl === "undefined") {
872
+ isControl = rollForControlGroup(variant.pz.control);
873
+ context.storage.updateData([
874
+ {
875
+ type: "setpersonalizecontrol",
876
+ data: {
877
+ personlizationName: name,
878
+ index: currentIndex,
879
+ control: isControl
880
+ }
881
+ }
882
+ ]);
883
+ }
884
+ if (isControl) {
885
+ variantToAdd = {
886
+ ...defaultVariants[0],
887
+ id: variant.id
888
+ };
889
+ }
890
+ }
891
+ personalized = personalized || typeof variantToAdd.pz !== "undefined";
892
+ results.push({
893
+ ...variantToAdd,
894
+ control: isControl
895
+ });
829
896
  }
830
897
  }
831
898
  return {
@@ -1072,16 +1139,19 @@ function parseScoreCookie(cookieValue) {
1072
1139
  if (!cookieValue)
1073
1140
  return;
1074
1141
  const types = cookieValue.split(TYPE_SEP);
1075
- if (types.length > 3)
1142
+ if (types.length > 5)
1076
1143
  return;
1077
- const [abTestData, sessionScores, visitorScores] = types;
1078
- return {
1144
+ const [abTestData, sessionScores, visitorScores, controlGroup, personalizeVariants] = types;
1145
+ const data = {
1079
1146
  // this is true since we're reading a cookie, which wouldn't exist if consent wasn't given
1080
1147
  consent: true,
1081
1148
  sessionScores: decodeCookieType(parseCookieType(sessionScores)),
1082
1149
  scores: decodeCookieType(parseCookieType(visitorScores)),
1083
- tests: parseCookieType(abTestData)
1150
+ tests: parseCookieType(abTestData),
1151
+ controlGroup: controlGroup === "1",
1152
+ personalizeVariants: decodePersonalizeVariants(personalizeVariants)
1084
1153
  };
1154
+ return data;
1085
1155
  }
1086
1156
  function parseCookieType(type) {
1087
1157
  if (!type) {
@@ -1105,9 +1175,48 @@ function serializeCookie(data) {
1105
1175
  return [
1106
1176
  serializeCookieType(data.tests),
1107
1177
  serializeCookieType(encodeCookieType(data.sessionScores)),
1108
- serializeCookieType(encodeCookieType(data.scores))
1178
+ serializeCookieType(encodeCookieType(data.scores)),
1179
+ data.controlGroup ? "1" : "0",
1180
+ serializePersonalizeVariants(data)
1109
1181
  ].join(TYPE_SEP);
1110
1182
  }
1183
+ function serializePersonalizeVariants({
1184
+ personalizeVariants
1185
+ }) {
1186
+ const data = {};
1187
+ if (typeof personalizeVariants === "object") {
1188
+ Object.keys(personalizeVariants).forEach((personalizationName) => {
1189
+ const results = [];
1190
+ const variants = personalizeVariants[personalizationName];
1191
+ variants.forEach((variant) => {
1192
+ results.push(`${variant.index}:${variant.control ? "1" : "0"}`);
1193
+ });
1194
+ data[personalizationName] = results.join(",");
1195
+ });
1196
+ }
1197
+ const serialized = serializeCookieType(data);
1198
+ return serialized;
1199
+ }
1200
+ function decodePersonalizeVariants(data) {
1201
+ const parsed = parseCookieType(data);
1202
+ const keys = Object.keys(parsed);
1203
+ if (!keys.length) {
1204
+ return void 0;
1205
+ }
1206
+ const results = {};
1207
+ Object.keys(parsed).forEach((k) => {
1208
+ const variants = parsed[k].split(",");
1209
+ const key = decodeURIComponent(k);
1210
+ results[key] = variants.map((variant) => {
1211
+ const [index, control] = variant.split(":");
1212
+ return {
1213
+ index: parseInt(index, 10),
1214
+ control: control === "1"
1215
+ };
1216
+ });
1217
+ });
1218
+ return results;
1219
+ }
1111
1220
  function encodeCookieType(type) {
1112
1221
  return Object.entries(type).reduce((acc, [key, value]) => {
1113
1222
  acc[key] = ntob(value);
@@ -1196,7 +1305,8 @@ var emptyVisitorData = () => ({
1196
1305
  tests: {},
1197
1306
  goals: {},
1198
1307
  consent: false,
1199
- controlGroup: false
1308
+ controlGroup: false,
1309
+ personalizeVariants: {}
1200
1310
  });
1201
1311
 
1202
1312
  // src/storage/VisitorDataStore.ts
@@ -1245,6 +1355,25 @@ function applyCommandsToData(commands, state, inControlGroup) {
1245
1355
  case "setcontrol":
1246
1356
  newData.controlGroup = command.data;
1247
1357
  break;
1358
+ case "setpersonalizecontrol":
1359
+ if (!newData.personalizeVariants) {
1360
+ newData.personalizeVariants = {};
1361
+ }
1362
+ if (!newData.personalizeVariants[command.data.personlizationName]) {
1363
+ newData.personalizeVariants[command.data.personlizationName] = [];
1364
+ }
1365
+ const existingDef = newData.personalizeVariants[command.data.personlizationName].find(
1366
+ (i) => i.index === command.data.index
1367
+ );
1368
+ if (!existingDef) {
1369
+ newData.personalizeVariants[command.data.personlizationName].push({
1370
+ index: command.data.index,
1371
+ control: command.data.control
1372
+ });
1373
+ } else {
1374
+ console.warn("Overwriting existing control group definition is not allowed");
1375
+ }
1376
+ break;
1248
1377
  case "setgoal":
1249
1378
  newData.goals = (_c = newData.goals) != null ? _c : {};
1250
1379
  newData.goals[command.data.goal] = true;
@@ -1627,7 +1756,7 @@ var Context = class {
1627
1756
  * will NOT result in a recomputation of signal state.
1628
1757
  */
1629
1758
  async update(newData) {
1630
- var _a, _b, _c;
1759
+ var _a, _b, _c, _d;
1631
1760
  const commands = [];
1632
1761
  const newServerSideTests = {};
1633
1762
  if ((_a = __privateGet(this, _serverTransitionState)) == null ? void 0 : _a.quirks) {
@@ -1657,6 +1786,17 @@ var Context = class {
1657
1786
  );
1658
1787
  }
1659
1788
  }
1789
+ if ((_c = __privateGet(this, _serverTransitionState)) == null ? void 0 : _c.personalizeVariants) {
1790
+ Object.keys(__privateGet(this, _serverTransitionState).personalizeVariants).forEach((personalizationName) => {
1791
+ const variants = __privateGet(this, _serverTransitionState).personalizeVariants[personalizationName];
1792
+ variants.forEach((e) => {
1793
+ commands.push({
1794
+ type: "setpersonalizecontrol",
1795
+ data: { personlizationName: personalizationName, index: e.index, control: e.control }
1796
+ });
1797
+ });
1798
+ });
1799
+ }
1660
1800
  try {
1661
1801
  __privateGet(this, _mitt3).emit("log", [
1662
1802
  "info",
@@ -1666,7 +1806,7 @@ var Context = class {
1666
1806
  ...newData,
1667
1807
  // need to convert url to string so it can be json serialized
1668
1808
  // to go over postMessage to chrome extension
1669
- url: (_c = newData.url) == null ? void 0 : _c.toString()
1809
+ url: (_d = newData.url) == null ? void 0 : _d.toString()
1670
1810
  }
1671
1811
  ]);
1672
1812
  if (newData.quirks) {
@@ -1757,6 +1897,13 @@ var Context = class {
1757
1897
  }
1758
1898
  ]);
1759
1899
  }
1900
+ getPersonalizeVariantControl(name, index) {
1901
+ var _a, _b, _c;
1902
+ const source = (_b = (_a = __privateGet(this, _serverTransitionState)) == null ? void 0 : _a.personalizeVariants) != null ? _b : this.storage.data.personalizeVariants;
1903
+ const variants = (_c = source == null ? void 0 : source[name]) != null ? _c : [];
1904
+ const variant = variants.find((v) => v.index === index);
1905
+ return variant == null ? void 0 : variant.control;
1906
+ }
1760
1907
  /**
1761
1908
  * Writes a message to the Context log sink.
1762
1909
  * Used by Uniform internal SDK; not intended for public use.
@@ -1799,10 +1946,10 @@ var Context = class {
1799
1946
  const previousPlacement = __privateGet(this, _pzCache)[options.name];
1800
1947
  const eventData = {
1801
1948
  name: options.name,
1802
- variantIds: value.variations.map((variation) => {
1803
- var _a;
1804
- return (_a = variation.id) != null ? _a : "Unknown";
1805
- }),
1949
+ variantIds: value.variations.map((variation) => ({
1950
+ id: variation.id || "Unknown",
1951
+ control: variation.control
1952
+ })),
1806
1953
  control: this.storage.data.controlGroup,
1807
1954
  changed: true
1808
1955
  };
@@ -1839,7 +1986,8 @@ var Context = class {
1839
1986
  const transitionState = {
1840
1987
  quirks: this.storage.data.quirks,
1841
1988
  ssv: __privateGet(this, _scores),
1842
- tests: {}
1989
+ tests: {},
1990
+ personalizeVariants: this.storage.data.personalizeVariants
1843
1991
  };
1844
1992
  const allTests = this.storage.data.tests;
1845
1993
  Object.entries(allTests).map(([testName, testValue]) => {
@@ -2539,7 +2687,7 @@ var createInsightsClient = ({ endpoint }) => {
2539
2687
  return sendMessage(message);
2540
2688
  },
2541
2689
  personalizationResult: async (options) => {
2542
- const messages = options.variantIds.map((variantId) => {
2690
+ const messages = options.variantIds.map((variant) => {
2543
2691
  const message = {
2544
2692
  action: "personalization_result",
2545
2693
  version: "1",
@@ -2550,8 +2698,8 @@ var createInsightsClient = ({ endpoint }) => {
2550
2698
  payload: {
2551
2699
  ...getBasePayload(),
2552
2700
  name: options.name,
2553
- variantId,
2554
- control: options.control,
2701
+ variantId: variant.id,
2702
+ control: variant.control || options.control,
2555
2703
  changed: options.changed
2556
2704
  }
2557
2705
  };
package/dist/index.mjs CHANGED
@@ -522,6 +522,15 @@ var SignalInstance = class {
522
522
  _evaluator = new WeakMap();
523
523
  _onLogMessage = new WeakMap();
524
524
 
525
+ // src/manifest/utils/control.ts
526
+ var rollForControlGroup = (value) => {
527
+ let control = value;
528
+ if (control >= 1) {
529
+ control = control / 100;
530
+ }
531
+ return Math.random() < control;
532
+ };
533
+
525
534
  // src/manifest/ManifestInstance.ts
526
535
  var _mf, _signalInstances, _goalEvaluators, _onLogMessage2;
527
536
  var ManifestInstance = class {
@@ -556,12 +565,8 @@ var ManifestInstance = class {
556
565
  __privateSet(this, _onLogMessage2, onLogMessage);
557
566
  }
558
567
  rollForControlGroup() {
559
- var _a, _b;
560
- let control = (_b = (_a = __privateGet(this, _mf).pz) == null ? void 0 : _a.control) != null ? _b : 0;
561
- if (control >= 1) {
562
- control = control / 100;
563
- }
564
- return Math.random() < control;
568
+ var _a;
569
+ return rollForControlGroup(((_a = __privateGet(this, _mf).pz) == null ? void 0 : _a.control) || 0);
565
570
  }
566
571
  getTest(name) {
567
572
  var _a;
@@ -643,7 +648,30 @@ function evaluateDimensionMatch(crit, vec, onLogMessage) {
643
648
  var _a, _b;
644
649
  const { op, l: lhs } = crit;
645
650
  const lhsScore = (_a = vec[lhs]) != null ? _a : 0;
646
- if (op === "+") {
651
+ if (op === "^") {
652
+ const [cat] = lhs.split(ENR_SEPARATOR);
653
+ let topVectorName = void 0;
654
+ let topScore = 0;
655
+ Object.keys(vec).forEach((vectorName) => {
656
+ if (vectorName.startsWith(`${cat}${ENR_SEPARATOR}`)) {
657
+ const score = vec[vectorName];
658
+ if (score > topScore) {
659
+ topVectorName = vectorName;
660
+ topScore = score;
661
+ }
662
+ }
663
+ });
664
+ const result = topVectorName === lhs;
665
+ onLogMessage == null ? void 0 : onLogMessage([
666
+ "info",
667
+ 302,
668
+ {
669
+ matched: result,
670
+ description: `${crit.l} has the highest score in the category`
671
+ }
672
+ ]);
673
+ return result;
674
+ } else if (op === "+") {
647
675
  const result = Math.max(...Object.values(vec)) === lhsScore && lhsScore > 0;
648
676
  onLogMessage == null ? void 0 : onLogMessage([
649
677
  "info",
@@ -730,31 +758,70 @@ function personalizeVariations({
730
758
  take = 1,
731
759
  onLogMessage
732
760
  }) {
733
- var _a, _b, _c;
761
+ var _a, _b, _c, _d;
734
762
  onLogMessage == null ? void 0 : onLogMessage(["info", 300, "GROUP", { name, take }]);
735
763
  try {
736
764
  const control = (_a = context.storage.data.controlGroup) != null ? _a : false;
737
765
  const results = [];
738
766
  let personalized = false;
739
767
  const scores = context.scores;
768
+ let index = 0;
769
+ const defaultVariants = [];
770
+ for (const variant of variations) {
771
+ if (!((_b = variant.pz) == null ? void 0 : _b.crit.length)) {
772
+ defaultVariants.push(variant);
773
+ }
774
+ }
740
775
  for (const variant of variations) {
776
+ const currentIndex = index++;
741
777
  if (results.length === take) {
742
778
  break;
743
779
  }
744
- if (!((_b = variant.pz) == null ? void 0 : _b.crit.length)) {
745
- onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, op: (_c = variant.pz) == null ? void 0 : _c.op }]);
780
+ if (!((_c = variant.pz) == null ? void 0 : _c.crit.length)) {
781
+ onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, op: (_d = variant.pz) == null ? void 0 : _d.op }]);
746
782
  onLogMessage == null ? void 0 : onLogMessage(["info", 302, { matched: true, description: "default variation" }]);
747
783
  onLogMessage == null ? void 0 : onLogMessage(["info", 303, true]);
748
784
  onLogMessage == null ? void 0 : onLogMessage(["info", 301, "ENDGROUP"]);
749
- results.push(variant);
785
+ results.push({
786
+ ...variant,
787
+ control: false
788
+ });
750
789
  continue;
751
790
  }
752
791
  if (control) {
753
792
  continue;
754
793
  }
755
794
  if (evaluateVariantMatch(variant.id, variant.pz, scores, onLogMessage)) {
756
- personalized = true;
757
- results.push(variant);
795
+ let variantToAdd = variant;
796
+ let isControl = false;
797
+ const isDefault = defaultVariants.find((v) => v.id === variant.id);
798
+ if (take === 1 && !isDefault && defaultVariants.length && typeof variant.pz.control === "number") {
799
+ isControl = context.getPersonalizeVariantControl(name, currentIndex);
800
+ if (typeof isControl === "undefined") {
801
+ isControl = rollForControlGroup(variant.pz.control);
802
+ context.storage.updateData([
803
+ {
804
+ type: "setpersonalizecontrol",
805
+ data: {
806
+ personlizationName: name,
807
+ index: currentIndex,
808
+ control: isControl
809
+ }
810
+ }
811
+ ]);
812
+ }
813
+ if (isControl) {
814
+ variantToAdd = {
815
+ ...defaultVariants[0],
816
+ id: variant.id
817
+ };
818
+ }
819
+ }
820
+ personalized = personalized || typeof variantToAdd.pz !== "undefined";
821
+ results.push({
822
+ ...variantToAdd,
823
+ control: isControl
824
+ });
758
825
  }
759
826
  }
760
827
  return {
@@ -1001,16 +1068,19 @@ function parseScoreCookie(cookieValue) {
1001
1068
  if (!cookieValue)
1002
1069
  return;
1003
1070
  const types = cookieValue.split(TYPE_SEP);
1004
- if (types.length > 3)
1071
+ if (types.length > 5)
1005
1072
  return;
1006
- const [abTestData, sessionScores, visitorScores] = types;
1007
- return {
1073
+ const [abTestData, sessionScores, visitorScores, controlGroup, personalizeVariants] = types;
1074
+ const data = {
1008
1075
  // this is true since we're reading a cookie, which wouldn't exist if consent wasn't given
1009
1076
  consent: true,
1010
1077
  sessionScores: decodeCookieType(parseCookieType(sessionScores)),
1011
1078
  scores: decodeCookieType(parseCookieType(visitorScores)),
1012
- tests: parseCookieType(abTestData)
1079
+ tests: parseCookieType(abTestData),
1080
+ controlGroup: controlGroup === "1",
1081
+ personalizeVariants: decodePersonalizeVariants(personalizeVariants)
1013
1082
  };
1083
+ return data;
1014
1084
  }
1015
1085
  function parseCookieType(type) {
1016
1086
  if (!type) {
@@ -1034,9 +1104,48 @@ function serializeCookie(data) {
1034
1104
  return [
1035
1105
  serializeCookieType(data.tests),
1036
1106
  serializeCookieType(encodeCookieType(data.sessionScores)),
1037
- serializeCookieType(encodeCookieType(data.scores))
1107
+ serializeCookieType(encodeCookieType(data.scores)),
1108
+ data.controlGroup ? "1" : "0",
1109
+ serializePersonalizeVariants(data)
1038
1110
  ].join(TYPE_SEP);
1039
1111
  }
1112
+ function serializePersonalizeVariants({
1113
+ personalizeVariants
1114
+ }) {
1115
+ const data = {};
1116
+ if (typeof personalizeVariants === "object") {
1117
+ Object.keys(personalizeVariants).forEach((personalizationName) => {
1118
+ const results = [];
1119
+ const variants = personalizeVariants[personalizationName];
1120
+ variants.forEach((variant) => {
1121
+ results.push(`${variant.index}:${variant.control ? "1" : "0"}`);
1122
+ });
1123
+ data[personalizationName] = results.join(",");
1124
+ });
1125
+ }
1126
+ const serialized = serializeCookieType(data);
1127
+ return serialized;
1128
+ }
1129
+ function decodePersonalizeVariants(data) {
1130
+ const parsed = parseCookieType(data);
1131
+ const keys = Object.keys(parsed);
1132
+ if (!keys.length) {
1133
+ return void 0;
1134
+ }
1135
+ const results = {};
1136
+ Object.keys(parsed).forEach((k) => {
1137
+ const variants = parsed[k].split(",");
1138
+ const key = decodeURIComponent(k);
1139
+ results[key] = variants.map((variant) => {
1140
+ const [index, control] = variant.split(":");
1141
+ return {
1142
+ index: parseInt(index, 10),
1143
+ control: control === "1"
1144
+ };
1145
+ });
1146
+ });
1147
+ return results;
1148
+ }
1040
1149
  function encodeCookieType(type) {
1041
1150
  return Object.entries(type).reduce((acc, [key, value]) => {
1042
1151
  acc[key] = ntob(value);
@@ -1125,7 +1234,8 @@ var emptyVisitorData = () => ({
1125
1234
  tests: {},
1126
1235
  goals: {},
1127
1236
  consent: false,
1128
- controlGroup: false
1237
+ controlGroup: false,
1238
+ personalizeVariants: {}
1129
1239
  });
1130
1240
 
1131
1241
  // src/storage/VisitorDataStore.ts
@@ -1174,6 +1284,25 @@ function applyCommandsToData(commands, state, inControlGroup) {
1174
1284
  case "setcontrol":
1175
1285
  newData.controlGroup = command.data;
1176
1286
  break;
1287
+ case "setpersonalizecontrol":
1288
+ if (!newData.personalizeVariants) {
1289
+ newData.personalizeVariants = {};
1290
+ }
1291
+ if (!newData.personalizeVariants[command.data.personlizationName]) {
1292
+ newData.personalizeVariants[command.data.personlizationName] = [];
1293
+ }
1294
+ const existingDef = newData.personalizeVariants[command.data.personlizationName].find(
1295
+ (i) => i.index === command.data.index
1296
+ );
1297
+ if (!existingDef) {
1298
+ newData.personalizeVariants[command.data.personlizationName].push({
1299
+ index: command.data.index,
1300
+ control: command.data.control
1301
+ });
1302
+ } else {
1303
+ console.warn("Overwriting existing control group definition is not allowed");
1304
+ }
1305
+ break;
1177
1306
  case "setgoal":
1178
1307
  newData.goals = (_c = newData.goals) != null ? _c : {};
1179
1308
  newData.goals[command.data.goal] = true;
@@ -1556,7 +1685,7 @@ var Context = class {
1556
1685
  * will NOT result in a recomputation of signal state.
1557
1686
  */
1558
1687
  async update(newData) {
1559
- var _a, _b, _c;
1688
+ var _a, _b, _c, _d;
1560
1689
  const commands = [];
1561
1690
  const newServerSideTests = {};
1562
1691
  if ((_a = __privateGet(this, _serverTransitionState)) == null ? void 0 : _a.quirks) {
@@ -1586,6 +1715,17 @@ var Context = class {
1586
1715
  );
1587
1716
  }
1588
1717
  }
1718
+ if ((_c = __privateGet(this, _serverTransitionState)) == null ? void 0 : _c.personalizeVariants) {
1719
+ Object.keys(__privateGet(this, _serverTransitionState).personalizeVariants).forEach((personalizationName) => {
1720
+ const variants = __privateGet(this, _serverTransitionState).personalizeVariants[personalizationName];
1721
+ variants.forEach((e) => {
1722
+ commands.push({
1723
+ type: "setpersonalizecontrol",
1724
+ data: { personlizationName: personalizationName, index: e.index, control: e.control }
1725
+ });
1726
+ });
1727
+ });
1728
+ }
1589
1729
  try {
1590
1730
  __privateGet(this, _mitt3).emit("log", [
1591
1731
  "info",
@@ -1595,7 +1735,7 @@ var Context = class {
1595
1735
  ...newData,
1596
1736
  // need to convert url to string so it can be json serialized
1597
1737
  // to go over postMessage to chrome extension
1598
- url: (_c = newData.url) == null ? void 0 : _c.toString()
1738
+ url: (_d = newData.url) == null ? void 0 : _d.toString()
1599
1739
  }
1600
1740
  ]);
1601
1741
  if (newData.quirks) {
@@ -1686,6 +1826,13 @@ var Context = class {
1686
1826
  }
1687
1827
  ]);
1688
1828
  }
1829
+ getPersonalizeVariantControl(name, index) {
1830
+ var _a, _b, _c;
1831
+ const source = (_b = (_a = __privateGet(this, _serverTransitionState)) == null ? void 0 : _a.personalizeVariants) != null ? _b : this.storage.data.personalizeVariants;
1832
+ const variants = (_c = source == null ? void 0 : source[name]) != null ? _c : [];
1833
+ const variant = variants.find((v) => v.index === index);
1834
+ return variant == null ? void 0 : variant.control;
1835
+ }
1689
1836
  /**
1690
1837
  * Writes a message to the Context log sink.
1691
1838
  * Used by Uniform internal SDK; not intended for public use.
@@ -1728,10 +1875,10 @@ var Context = class {
1728
1875
  const previousPlacement = __privateGet(this, _pzCache)[options.name];
1729
1876
  const eventData = {
1730
1877
  name: options.name,
1731
- variantIds: value.variations.map((variation) => {
1732
- var _a;
1733
- return (_a = variation.id) != null ? _a : "Unknown";
1734
- }),
1878
+ variantIds: value.variations.map((variation) => ({
1879
+ id: variation.id || "Unknown",
1880
+ control: variation.control
1881
+ })),
1735
1882
  control: this.storage.data.controlGroup,
1736
1883
  changed: true
1737
1884
  };
@@ -1768,7 +1915,8 @@ var Context = class {
1768
1915
  const transitionState = {
1769
1916
  quirks: this.storage.data.quirks,
1770
1917
  ssv: __privateGet(this, _scores),
1771
- tests: {}
1918
+ tests: {},
1919
+ personalizeVariants: this.storage.data.personalizeVariants
1772
1920
  };
1773
1921
  const allTests = this.storage.data.tests;
1774
1922
  Object.entries(allTests).map(([testName, testValue]) => {
@@ -2468,7 +2616,7 @@ var createInsightsClient = ({ endpoint }) => {
2468
2616
  return sendMessage(message);
2469
2617
  },
2470
2618
  personalizationResult: async (options) => {
2471
- const messages = options.variantIds.map((variantId) => {
2619
+ const messages = options.variantIds.map((variant) => {
2472
2620
  const message = {
2473
2621
  action: "personalization_result",
2474
2622
  version: "1",
@@ -2479,8 +2627,8 @@ var createInsightsClient = ({ endpoint }) => {
2479
2627
  payload: {
2480
2628
  ...getBasePayload(),
2481
2629
  name: options.name,
2482
- variantId,
2483
- control: options.control,
2630
+ variantId: variant.id,
2631
+ control: variant.control || options.control,
2484
2632
  changed: options.changed
2485
2633
  }
2486
2634
  };
@@ -20,6 +20,13 @@ type EnrichmentData = {
20
20
  /** Strength value (amount of score added when viewing content) */
21
21
  str: number;
22
22
  };
23
+ type PersonalizeControlVariant = {
24
+ index: number;
25
+ control: boolean;
26
+ };
27
+ type PersonalizeVariants = {
28
+ [key: string]: PersonalizeControlVariant[];
29
+ };
23
30
  /** An event that has occurred (i.e. an analytics track) which may trigger an Event signal */
24
31
  type EventData = {
25
32
  /** The event name that has been fired */
@@ -62,6 +69,10 @@ type VisitorData = {
62
69
  * based on the control group size.
63
70
  */
64
71
  controlGroup?: boolean;
72
+ /**
73
+ * Records of personalized variants that have been shown to the visitor and their control group status
74
+ */
75
+ personalizeVariants?: PersonalizeVariants;
65
76
  };
66
77
  declare const emptyVisitorData: () => VisitorData;
67
78
  /**
@@ -90,7 +101,7 @@ type StorageCommand<TID extends string = string, TData = unknown> = {
90
101
  data: TData;
91
102
  };
92
103
  /** Commands that can be issued to alter the storage of Uniform Context data */
93
- type StorageCommands = ModifyScoreCommand | ModifySessionScoreCommand | SetConsentCommand | SetQuirkCommand | SetTestCommand | IdentifyCommand | SetControlGroupCommand | SetGoalCommand;
104
+ type StorageCommands = ModifyScoreCommand | ModifySessionScoreCommand | SetConsentCommand | SetQuirkCommand | SetTestCommand | IdentifyCommand | SetControlGroupCommand | SetPersonalizeVariantControlCommand | SetGoalCommand;
94
105
  /**
95
106
  * Converts the goal specified.
96
107
  */
@@ -142,11 +153,16 @@ type IdentifyCommand = StorageCommand<'identify', {
142
153
  * this command is intended mostly for diagnostics and testing purposes.
143
154
  */
144
155
  type SetControlGroupCommand = StorageCommand<'setcontrol', boolean>;
156
+ type SetPersonalizeVariantControlCommand = StorageCommand<'setpersonalizecontrol', {
157
+ personlizationName: string;
158
+ index: number;
159
+ control: boolean;
160
+ }>;
145
161
 
146
162
  type TransitionDataStoreOptions = {
147
163
  initialData?: Partial<VisitorData>;
148
164
  };
149
- type ServerToClientTransitionState = Pick<Partial<VisitorData>, 'quirks' | 'tests'> & {
165
+ type ServerToClientTransitionState = Pick<Partial<VisitorData>, 'quirks' | 'tests' | 'personalizeVariants'> & {
150
166
  /**
151
167
  * Server Score Vector - the resultant scores _on the server side_ after the server/edge render completes
152
168
  * Note that the client side does not trust these scores; they are only used until it's done with initial
@@ -911,6 +927,10 @@ type VariantMatchCriteria = {
911
927
  * Name of the variant for analytics tracking.
912
928
  */
913
929
  name?: string;
930
+ /**
931
+ * Control group percentage for the variant.
932
+ */
933
+ control?: number;
914
934
  };
915
935
  type DimensionMatch = {
916
936
  /**
@@ -921,7 +941,8 @@ type DimensionMatch = {
921
941
  /**
922
942
  * Operator of the match expression
923
943
  * Whole-vector (RHS only) operators - these do not require a `r` or `rDim` set:
924
- * +: `l` is the strongest dimension in the score vector
944
+ * +: `l` is the strongest dimension in the score vector across all categories
945
+ * ^: `l` is the strongest dimension in the specified category
925
946
  * -: `l` is the weakest dimension in the score vector. This does not match if the dimension has no score at all.
926
947
  *
927
948
  * Comparison operators:
@@ -932,7 +953,7 @@ type DimensionMatch = {
932
953
  * =: `l` is equal to the right hand side expression
933
954
  * !=: `l` is not equal to the right hand side expression
934
955
  */
935
- op: '+' | '-' | '>' | '>=' | '<' | '<=' | '=' | '!=';
956
+ op: '+' | '-' | '>' | '>=' | '<' | '<=' | '=' | '!=' | '^';
936
957
  /**
937
958
  * Right hand side of the match expression (not required for op = + or - which have no right side)
938
959
  * This value is treated as a constant value, if it is present. If it's a string, it is parsed to an integer.
@@ -966,7 +987,9 @@ type PersonalizedResult<TVariant> = {
966
987
  /** Whether or not this result contains a personalized result */
967
988
  personalized: boolean;
968
989
  /** Matching variations */
969
- variations: Array<TVariant>;
990
+ variations: Array<TVariant & {
991
+ control: boolean;
992
+ }>;
970
993
  };
971
994
  /** Defines the shape of a A/B test variant */
972
995
  type TestVariant = {
@@ -1044,7 +1067,10 @@ type PersonalizationEvent = {
1044
1067
  /** Name of the personalized placement */
1045
1068
  name: string;
1046
1069
  /** Selected variant ID(s) */
1047
- variantIds: string[];
1070
+ variantIds: {
1071
+ id: string;
1072
+ control: boolean;
1073
+ }[];
1048
1074
  /** Whether the user was part of the control group (and did not receive any personalization) */
1049
1075
  control: boolean | undefined;
1050
1076
  /**
@@ -1143,6 +1169,7 @@ declare class Context implements ContextInstance {
1143
1169
  getTestVariantId(testName: string): string | null | undefined;
1144
1170
  /** use test() instead */
1145
1171
  setTestVariantId(testName: string, variantId: string): void;
1172
+ getPersonalizeVariantControl(name: string, index: number): boolean | undefined;
1146
1173
  /**
1147
1174
  * Writes a message to the Context log sink.
1148
1175
  * Used by Uniform internal SDK; not intended for public use.
@@ -1233,4 +1260,4 @@ declare global {
1233
1260
  }
1234
1261
  }
1235
1262
 
1236
- export { type Goal as $, type AggregateDimension as A, type MessageFunc as B, type ContextPlugin as C, type DecayFunction as D, type LogMessageSingle as E, type LogMessageGroup as F, ManifestInstance as G, GroupCriteriaEvaluator as H, type CriteriaEvaluatorResult as I, type CriteriaEvaluatorParameters as J, type SignalData as K, type LogDrain as L, type MessageCategory as M, type ManifestV2 as N, type OutputSeverity as O, type PersonalizationEvent as P, type PersonalizationManifest as Q, type Signal as R, type ScoreVector as S, TransitionDataStore as T, type SignalCriteriaGroup as U, type VisitorData as V, type SignalCriteria as W, type EnrichmentCategory as X, type NumberMatch as Y, type TestDefinition as Z, type AggregateDimensionInput as _, type StorageCommands as a, type SignalGoal as a0, type EnrichmentGoal as a1, type QuirkGoal as a2, type TestOptions as a3, testVariations as a4, type DimensionMatch as a5, type PersonalizeOptions as a6, personalizeVariations as a7, type BehaviorTag as a8, type PersonalizedVariant as a9, type ContextStateUpdate as aA, type GoalStateUpdate as aB, type paths as aC, type PersonalizedResult as aa, type TestVariant as ab, type TestResult as ac, type StorageCommand as ad, type SetGoalCommand as ae, type ModifyScoreCommand as af, type ModifySessionScoreCommand as ag, type SetConsentCommand as ah, type SetQuirkCommand as ai, type SetTestCommand as aj, type IdentifyCommand as ak, type SetControlGroupCommand as al, type ServerToClientTransitionState as am, SERVER_STATE_ID as an, type TransitionDataStoreEvents as ao, type DecayOptions as ap, type VisitorDataStoreOptions as aq, type VisitorDataStoreEvents as ar, VisitorDataStore as as, type Quirks as at, type Tests as au, type Goals as av, type EnrichmentData as aw, type EventData as ax, emptyVisitorData as ay, type ContextState as az, type TransitionDataStoreOptions as b, type CriteriaEvaluator as c, type StringMatch as d, type VariantMatchCriteria as e, type LogMessage as f, type DevToolsEvents as g, CONTEXTUAL_EDITING_TEST_NAME as h, CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID as i, type ContextOptions as j, type TestEvent as k, type ContextEvents as l, type ContextInstance as m, Context as n, type DevToolsUiVersion as o, type DevToolsState as p, type DevToolsActions as q, type DevToolsEvent as r, type DevToolsLogEvent as s, type DevToolsDataEvent as t, type DevToolsHelloEvent as u, type DevToolsUpdateEvent as v, type DevToolsRawCommandsEvent as w, type DevToolsForgetEvent as x, type LogMessages as y, type Severity as z };
1263
+ export { type Goal as $, type AggregateDimension as A, type MessageFunc as B, type ContextPlugin as C, type DecayFunction as D, type LogMessageSingle as E, type LogMessageGroup as F, ManifestInstance as G, GroupCriteriaEvaluator as H, type CriteriaEvaluatorResult as I, type CriteriaEvaluatorParameters as J, type SignalData as K, type LogDrain as L, type MessageCategory as M, type ManifestV2 as N, type OutputSeverity as O, type PersonalizationEvent as P, type PersonalizationManifest as Q, type Signal as R, type ScoreVector as S, TransitionDataStore as T, type SignalCriteriaGroup as U, type VisitorData as V, type SignalCriteria as W, type EnrichmentCategory as X, type NumberMatch as Y, type TestDefinition as Z, type AggregateDimensionInput as _, type StorageCommands as a, type SignalGoal as a0, type EnrichmentGoal as a1, type QuirkGoal as a2, type TestOptions as a3, testVariations as a4, type DimensionMatch as a5, type PersonalizeOptions as a6, personalizeVariations as a7, type BehaviorTag as a8, type PersonalizedVariant as a9, type EventData as aA, emptyVisitorData as aB, type ContextState as aC, type ContextStateUpdate as aD, type GoalStateUpdate as aE, type paths as aF, type PersonalizedResult as aa, type TestVariant as ab, type TestResult as ac, type StorageCommand as ad, type SetGoalCommand as ae, type ModifyScoreCommand as af, type ModifySessionScoreCommand as ag, type SetConsentCommand as ah, type SetQuirkCommand as ai, type SetTestCommand as aj, type IdentifyCommand as ak, type SetControlGroupCommand as al, type SetPersonalizeVariantControlCommand as am, type ServerToClientTransitionState as an, SERVER_STATE_ID as ao, type TransitionDataStoreEvents as ap, type DecayOptions as aq, type VisitorDataStoreOptions as ar, type VisitorDataStoreEvents as as, VisitorDataStore as at, type Quirks as au, type Tests as av, type Goals as aw, type EnrichmentData as ax, type PersonalizeControlVariant as ay, type PersonalizeVariants as az, type TransitionDataStoreOptions as b, type CriteriaEvaluator as c, type StringMatch as d, type VariantMatchCriteria as e, type LogMessage as f, type DevToolsEvents as g, CONTEXTUAL_EDITING_TEST_NAME as h, CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID as i, type ContextOptions as j, type TestEvent as k, type ContextEvents as l, type ContextInstance as m, Context as n, type DevToolsUiVersion as o, type DevToolsState as p, type DevToolsActions as q, type DevToolsEvent as r, type DevToolsLogEvent as s, type DevToolsDataEvent as t, type DevToolsHelloEvent as u, type DevToolsUpdateEvent as v, type DevToolsRawCommandsEvent as w, type DevToolsForgetEvent as x, type LogMessages as y, type Severity as z };
@@ -20,6 +20,13 @@ type EnrichmentData = {
20
20
  /** Strength value (amount of score added when viewing content) */
21
21
  str: number;
22
22
  };
23
+ type PersonalizeControlVariant = {
24
+ index: number;
25
+ control: boolean;
26
+ };
27
+ type PersonalizeVariants = {
28
+ [key: string]: PersonalizeControlVariant[];
29
+ };
23
30
  /** An event that has occurred (i.e. an analytics track) which may trigger an Event signal */
24
31
  type EventData = {
25
32
  /** The event name that has been fired */
@@ -62,6 +69,10 @@ type VisitorData = {
62
69
  * based on the control group size.
63
70
  */
64
71
  controlGroup?: boolean;
72
+ /**
73
+ * Records of personalized variants that have been shown to the visitor and their control group status
74
+ */
75
+ personalizeVariants?: PersonalizeVariants;
65
76
  };
66
77
  declare const emptyVisitorData: () => VisitorData;
67
78
  /**
@@ -90,7 +101,7 @@ type StorageCommand<TID extends string = string, TData = unknown> = {
90
101
  data: TData;
91
102
  };
92
103
  /** Commands that can be issued to alter the storage of Uniform Context data */
93
- type StorageCommands = ModifyScoreCommand | ModifySessionScoreCommand | SetConsentCommand | SetQuirkCommand | SetTestCommand | IdentifyCommand | SetControlGroupCommand | SetGoalCommand;
104
+ type StorageCommands = ModifyScoreCommand | ModifySessionScoreCommand | SetConsentCommand | SetQuirkCommand | SetTestCommand | IdentifyCommand | SetControlGroupCommand | SetPersonalizeVariantControlCommand | SetGoalCommand;
94
105
  /**
95
106
  * Converts the goal specified.
96
107
  */
@@ -142,11 +153,16 @@ type IdentifyCommand = StorageCommand<'identify', {
142
153
  * this command is intended mostly for diagnostics and testing purposes.
143
154
  */
144
155
  type SetControlGroupCommand = StorageCommand<'setcontrol', boolean>;
156
+ type SetPersonalizeVariantControlCommand = StorageCommand<'setpersonalizecontrol', {
157
+ personlizationName: string;
158
+ index: number;
159
+ control: boolean;
160
+ }>;
145
161
 
146
162
  type TransitionDataStoreOptions = {
147
163
  initialData?: Partial<VisitorData>;
148
164
  };
149
- type ServerToClientTransitionState = Pick<Partial<VisitorData>, 'quirks' | 'tests'> & {
165
+ type ServerToClientTransitionState = Pick<Partial<VisitorData>, 'quirks' | 'tests' | 'personalizeVariants'> & {
150
166
  /**
151
167
  * Server Score Vector - the resultant scores _on the server side_ after the server/edge render completes
152
168
  * Note that the client side does not trust these scores; they are only used until it's done with initial
@@ -911,6 +927,10 @@ type VariantMatchCriteria = {
911
927
  * Name of the variant for analytics tracking.
912
928
  */
913
929
  name?: string;
930
+ /**
931
+ * Control group percentage for the variant.
932
+ */
933
+ control?: number;
914
934
  };
915
935
  type DimensionMatch = {
916
936
  /**
@@ -921,7 +941,8 @@ type DimensionMatch = {
921
941
  /**
922
942
  * Operator of the match expression
923
943
  * Whole-vector (RHS only) operators - these do not require a `r` or `rDim` set:
924
- * +: `l` is the strongest dimension in the score vector
944
+ * +: `l` is the strongest dimension in the score vector across all categories
945
+ * ^: `l` is the strongest dimension in the specified category
925
946
  * -: `l` is the weakest dimension in the score vector. This does not match if the dimension has no score at all.
926
947
  *
927
948
  * Comparison operators:
@@ -932,7 +953,7 @@ type DimensionMatch = {
932
953
  * =: `l` is equal to the right hand side expression
933
954
  * !=: `l` is not equal to the right hand side expression
934
955
  */
935
- op: '+' | '-' | '>' | '>=' | '<' | '<=' | '=' | '!=';
956
+ op: '+' | '-' | '>' | '>=' | '<' | '<=' | '=' | '!=' | '^';
936
957
  /**
937
958
  * Right hand side of the match expression (not required for op = + or - which have no right side)
938
959
  * This value is treated as a constant value, if it is present. If it's a string, it is parsed to an integer.
@@ -966,7 +987,9 @@ type PersonalizedResult<TVariant> = {
966
987
  /** Whether or not this result contains a personalized result */
967
988
  personalized: boolean;
968
989
  /** Matching variations */
969
- variations: Array<TVariant>;
990
+ variations: Array<TVariant & {
991
+ control: boolean;
992
+ }>;
970
993
  };
971
994
  /** Defines the shape of a A/B test variant */
972
995
  type TestVariant = {
@@ -1044,7 +1067,10 @@ type PersonalizationEvent = {
1044
1067
  /** Name of the personalized placement */
1045
1068
  name: string;
1046
1069
  /** Selected variant ID(s) */
1047
- variantIds: string[];
1070
+ variantIds: {
1071
+ id: string;
1072
+ control: boolean;
1073
+ }[];
1048
1074
  /** Whether the user was part of the control group (and did not receive any personalization) */
1049
1075
  control: boolean | undefined;
1050
1076
  /**
@@ -1143,6 +1169,7 @@ declare class Context implements ContextInstance {
1143
1169
  getTestVariantId(testName: string): string | null | undefined;
1144
1170
  /** use test() instead */
1145
1171
  setTestVariantId(testName: string, variantId: string): void;
1172
+ getPersonalizeVariantControl(name: string, index: number): boolean | undefined;
1146
1173
  /**
1147
1174
  * Writes a message to the Context log sink.
1148
1175
  * Used by Uniform internal SDK; not intended for public use.
@@ -1233,4 +1260,4 @@ declare global {
1233
1260
  }
1234
1261
  }
1235
1262
 
1236
- export { type Goal as $, type AggregateDimension as A, type MessageFunc as B, type ContextPlugin as C, type DecayFunction as D, type LogMessageSingle as E, type LogMessageGroup as F, ManifestInstance as G, GroupCriteriaEvaluator as H, type CriteriaEvaluatorResult as I, type CriteriaEvaluatorParameters as J, type SignalData as K, type LogDrain as L, type MessageCategory as M, type ManifestV2 as N, type OutputSeverity as O, type PersonalizationEvent as P, type PersonalizationManifest as Q, type Signal as R, type ScoreVector as S, TransitionDataStore as T, type SignalCriteriaGroup as U, type VisitorData as V, type SignalCriteria as W, type EnrichmentCategory as X, type NumberMatch as Y, type TestDefinition as Z, type AggregateDimensionInput as _, type StorageCommands as a, type SignalGoal as a0, type EnrichmentGoal as a1, type QuirkGoal as a2, type TestOptions as a3, testVariations as a4, type DimensionMatch as a5, type PersonalizeOptions as a6, personalizeVariations as a7, type BehaviorTag as a8, type PersonalizedVariant as a9, type ContextStateUpdate as aA, type GoalStateUpdate as aB, type paths as aC, type PersonalizedResult as aa, type TestVariant as ab, type TestResult as ac, type StorageCommand as ad, type SetGoalCommand as ae, type ModifyScoreCommand as af, type ModifySessionScoreCommand as ag, type SetConsentCommand as ah, type SetQuirkCommand as ai, type SetTestCommand as aj, type IdentifyCommand as ak, type SetControlGroupCommand as al, type ServerToClientTransitionState as am, SERVER_STATE_ID as an, type TransitionDataStoreEvents as ao, type DecayOptions as ap, type VisitorDataStoreOptions as aq, type VisitorDataStoreEvents as ar, VisitorDataStore as as, type Quirks as at, type Tests as au, type Goals as av, type EnrichmentData as aw, type EventData as ax, emptyVisitorData as ay, type ContextState as az, type TransitionDataStoreOptions as b, type CriteriaEvaluator as c, type StringMatch as d, type VariantMatchCriteria as e, type LogMessage as f, type DevToolsEvents as g, CONTEXTUAL_EDITING_TEST_NAME as h, CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID as i, type ContextOptions as j, type TestEvent as k, type ContextEvents as l, type ContextInstance as m, Context as n, type DevToolsUiVersion as o, type DevToolsState as p, type DevToolsActions as q, type DevToolsEvent as r, type DevToolsLogEvent as s, type DevToolsDataEvent as t, type DevToolsHelloEvent as u, type DevToolsUpdateEvent as v, type DevToolsRawCommandsEvent as w, type DevToolsForgetEvent as x, type LogMessages as y, type Severity as z };
1263
+ export { type Goal as $, type AggregateDimension as A, type MessageFunc as B, type ContextPlugin as C, type DecayFunction as D, type LogMessageSingle as E, type LogMessageGroup as F, ManifestInstance as G, GroupCriteriaEvaluator as H, type CriteriaEvaluatorResult as I, type CriteriaEvaluatorParameters as J, type SignalData as K, type LogDrain as L, type MessageCategory as M, type ManifestV2 as N, type OutputSeverity as O, type PersonalizationEvent as P, type PersonalizationManifest as Q, type Signal as R, type ScoreVector as S, TransitionDataStore as T, type SignalCriteriaGroup as U, type VisitorData as V, type SignalCriteria as W, type EnrichmentCategory as X, type NumberMatch as Y, type TestDefinition as Z, type AggregateDimensionInput as _, type StorageCommands as a, type SignalGoal as a0, type EnrichmentGoal as a1, type QuirkGoal as a2, type TestOptions as a3, testVariations as a4, type DimensionMatch as a5, type PersonalizeOptions as a6, personalizeVariations as a7, type BehaviorTag as a8, type PersonalizedVariant as a9, type EventData as aA, emptyVisitorData as aB, type ContextState as aC, type ContextStateUpdate as aD, type GoalStateUpdate as aE, type paths as aF, type PersonalizedResult as aa, type TestVariant as ab, type TestResult as ac, type StorageCommand as ad, type SetGoalCommand as ae, type ModifyScoreCommand as af, type ModifySessionScoreCommand as ag, type SetConsentCommand as ah, type SetQuirkCommand as ai, type SetTestCommand as aj, type IdentifyCommand as ak, type SetControlGroupCommand as al, type SetPersonalizeVariantControlCommand as am, type ServerToClientTransitionState as an, SERVER_STATE_ID as ao, type TransitionDataStoreEvents as ap, type DecayOptions as aq, type VisitorDataStoreOptions as ar, type VisitorDataStoreEvents as as, VisitorDataStore as at, type Quirks as au, type Tests as av, type Goals as aw, type EnrichmentData as ax, type PersonalizeControlVariant as ay, type PersonalizeVariants as az, type TransitionDataStoreOptions as b, type CriteriaEvaluator as c, type StringMatch as d, type VariantMatchCriteria as e, type LogMessage as f, type DevToolsEvents as g, CONTEXTUAL_EDITING_TEST_NAME as h, CONTEXTUAL_EDITING_TEST_SELECTED_VARIANT_ID as i, type ContextOptions as j, type TestEvent as k, type ContextEvents as l, type ContextInstance as m, Context as n, type DevToolsUiVersion as o, type DevToolsState as p, type DevToolsActions as q, type DevToolsEvent as r, type DevToolsLogEvent as s, type DevToolsDataEvent as t, type DevToolsHelloEvent as u, type DevToolsUpdateEvent as v, type DevToolsRawCommandsEvent as w, type DevToolsForgetEvent as x, type LogMessages as y, type Severity as z };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@uniformdev/context",
3
- "version": "19.79.1-alpha.13+608a273da",
3
+ "version": "19.79.1-alpha.25+87cea1cd47",
4
4
  "description": "Uniform Context core package",
5
5
  "license": "SEE LICENSE IN LICENSE.txt",
6
6
  "main": "./dist/index.js",
@@ -68,5 +68,5 @@
68
68
  "publishConfig": {
69
69
  "access": "public"
70
70
  },
71
- "gitHead": "608a273da87aa432f17b2c8ce15d747dadab0cd1"
71
+ "gitHead": "87cea1cd47c691f7d32b876660d5f7c984bb1d87"
72
72
  }