@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 +1 -1
- package/dist/api/api.d.mts +1 -1
- package/dist/api/api.d.ts +1 -1
- package/dist/index.d.mts +2 -2
- package/dist/index.d.ts +2 -2
- package/dist/index.esm.js +177 -29
- package/dist/index.js +177 -29
- package/dist/index.mjs +177 -29
- package/dist/{types-8605X2cu.d.mts → types-D4Fs5NVm.d.mts} +34 -7
- package/dist/{types-8605X2cu.d.ts → types-D4Fs5NVm.d.ts} +34 -7
- package/package.json +2 -2
package/LICENSE.txt
CHANGED
@@ -1,2 +1,2 @@
|
|
1
|
-
©
|
1
|
+
© 2024 Uniform Systems, Inc. All Rights Reserved.
|
2
2
|
See details of Uniform Systems, Inc. Master Subscription Agreement here: https://uniform.dev/eula
|
package/dist/api/api.d.mts
CHANGED
package/dist/api/api.d.ts
CHANGED
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-
|
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,
|
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-
|
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,
|
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
|
560
|
-
|
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 (!((
|
745
|
-
onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, 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(
|
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
|
-
|
757
|
-
|
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 >
|
1071
|
+
if (types.length > 5)
|
1005
1072
|
return;
|
1006
|
-
const [abTestData, sessionScores, visitorScores] = types;
|
1007
|
-
|
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: (
|
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
|
-
|
1733
|
-
|
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((
|
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
|
631
|
-
|
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 (!((
|
816
|
-
onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, 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(
|
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
|
-
|
828
|
-
|
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 >
|
1142
|
+
if (types.length > 5)
|
1076
1143
|
return;
|
1077
|
-
const [abTestData, sessionScores, visitorScores] = types;
|
1078
|
-
|
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: (
|
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
|
-
|
1804
|
-
|
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((
|
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
|
560
|
-
|
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 (!((
|
745
|
-
onLogMessage == null ? void 0 : onLogMessage(["info", 301, "GROUP", { id: variant.id, 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(
|
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
|
-
|
757
|
-
|
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 >
|
1071
|
+
if (types.length > 5)
|
1005
1072
|
return;
|
1006
|
-
const [abTestData, sessionScores, visitorScores] = types;
|
1007
|
-
|
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: (
|
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
|
-
|
1733
|
-
|
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((
|
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:
|
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
|
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:
|
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
|
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.
|
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": "
|
71
|
+
"gitHead": "87cea1cd47c691f7d32b876660d5f7c984bb1d87"
|
72
72
|
}
|