@thoughtbot/superglue 1.0.2 → 2.0.0-alpha.1

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/.tool-versions CHANGED
@@ -1 +1 @@
1
- nodejs 18.20.4
1
+ nodejs 23.10.0
@@ -1,3 +1,3 @@
1
- export { a1 as MismatchedComponentError, a2 as remote, s as saveAndProcessPage, a3 as visit } from './index-BYr1PoYr.mjs';
1
+ export { a6 as MismatchedComponentError, a7 as remote, s as saveAndProcessPage, a8 as visit } from './index-MyfFLe4E.mjs';
2
2
  import '@reduxjs/toolkit';
3
3
  import 'history';
@@ -3,7 +3,7 @@ import {
3
3
  remote,
4
4
  saveAndProcessPage,
5
5
  visit
6
- } from "./chunk-7OMO5P27.mjs";
6
+ } from "./chunk-YS477OKK.mjs";
7
7
  export {
8
8
  MismatchedComponentError,
9
9
  remote,
@@ -1,3 +1,29 @@
1
+ var __create = Object.create;
2
+ var __defProp = Object.defineProperty;
3
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
+ var __getOwnPropNames = Object.getOwnPropertyNames;
5
+ var __getProtoOf = Object.getPrototypeOf;
6
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
7
+ var __commonJS = (cb, mod) => function __require() {
8
+ return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
9
+ };
10
+ var __copyProps = (to, from, except, desc) => {
11
+ if (from && typeof from === "object" || typeof from === "function") {
12
+ for (let key of __getOwnPropNames(from))
13
+ if (!__hasOwnProp.call(to, key) && key !== except)
14
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
+ }
16
+ return to;
17
+ };
18
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
19
+ // If the importer is in node compatibility mode or this is not an ESM
20
+ // file that has been converted to a CommonJS file using a Babel-
21
+ // compatible transform (i.e. "__esModule" has not been set), then set
22
+ // "default" to the CommonJS "module.exports" for node compatibility.
23
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
24
+ mod
25
+ ));
26
+
1
27
  // lib/utils/url.ts
2
28
  var FAKE_ORIGIN = "https://example.com";
3
29
  function pathQuery(url) {
@@ -179,7 +205,26 @@ var config = {
179
205
  maxPages: 20
180
206
  };
181
207
 
208
+ // lib/utils/limited_set.ts
209
+ var LimitedSet = class extends Set {
210
+ constructor(maxSize) {
211
+ super();
212
+ this.maxSize = maxSize;
213
+ }
214
+ add(value) {
215
+ if (this.size >= this.maxSize) {
216
+ const iterator = this.values();
217
+ const oldestValue = iterator.next().value;
218
+ this.delete(oldestValue);
219
+ }
220
+ super.add(value);
221
+ return this;
222
+ }
223
+ };
224
+
182
225
  // lib/utils/request.ts
226
+ import { v4 as uuidv4 } from "uuid";
227
+ var lastRequestIds = new LimitedSet(20);
183
228
  function isValidResponse(xhr) {
184
229
  return isValidContent(xhr) && !downloadingFile(xhr);
185
230
  }
@@ -235,6 +280,9 @@ function argsForFetch(getState, pathQuery2, {
235
280
  nextHeaders["x-requested-with"] = "XMLHttpRequest";
236
281
  nextHeaders["accept"] = "application/json";
237
282
  nextHeaders["x-superglue-request"] = "true";
283
+ const requestId = uuidv4();
284
+ lastRequestIds.add(requestId);
285
+ nextHeaders["X-Superglue-Request-Id"] = requestId;
238
286
  if (method != "GET" && method != "HEAD") {
239
287
  nextHeaders["content-type"] = "application/json";
240
288
  }
@@ -428,7 +476,6 @@ var handleGraft = createAction(
428
476
  var superglueError = createAction(
429
477
  "@@superglue/ERROR"
430
478
  );
431
- var updateFragments = createAction("@@superglue/UPDATE_FRAGMENTS");
432
479
  var copyPage = createAction(
433
480
  "@@superglue/COPY_PAGE"
434
481
  );
@@ -443,6 +490,53 @@ var beforeRemote = createAction("@@superglue/BEFORE_REMOTE");
443
490
  var setCSRFToken = createAction("@@superglue/SET_CSRF_TOKEN");
444
491
  var historyChange = createAction("@@superglue/HISTORY_CHANGE");
445
492
  var setActivePage = createAction("@@superglue/SET_ACTIVE_PAGE");
493
+ var handleFragmentGraft = createAction(
494
+ "@@superglue/HANDLE_FRAGMENT_GRAFT",
495
+ ({
496
+ fragmentId,
497
+ response
498
+ }) => {
499
+ return {
500
+ payload: {
501
+ response,
502
+ fragmentId
503
+ }
504
+ };
505
+ }
506
+ );
507
+ var saveFragment = createAction(
508
+ "@@superglue/SAVE_FRAGMENT",
509
+ ({ fragmentId, data }) => {
510
+ return {
511
+ payload: {
512
+ fragmentId,
513
+ data
514
+ }
515
+ };
516
+ }
517
+ );
518
+ var appendToFragment = createAction(
519
+ "@@superglue/APPEND_TO_FRAGMENT",
520
+ ({ data, fragmentId }) => {
521
+ return {
522
+ payload: {
523
+ data,
524
+ fragmentId
525
+ }
526
+ };
527
+ }
528
+ );
529
+ var prependToFragment = createAction(
530
+ "@@superglue/PREPEND_TO_FRAGMENT",
531
+ ({ data, fragmentId }) => {
532
+ return {
533
+ payload: {
534
+ data,
535
+ fragmentId
536
+ }
537
+ };
538
+ }
539
+ );
446
540
 
447
541
  // lib/action_creators/requests.ts
448
542
  function handleFetchErr(err, fetchArgs, dispatch) {
@@ -452,15 +546,18 @@ function handleFetchErr(err, fetchArgs, dispatch) {
452
546
  function buildMeta(pageKey, page, state, rsp, fetchArgs) {
453
547
  const { assets: prevAssets } = state;
454
548
  const { assets: nextAssets } = page;
455
- return {
549
+ const meta = {
456
550
  pageKey,
457
551
  page,
458
552
  redirected: rsp.redirected,
459
553
  rsp,
460
554
  fetchArgs,
461
- componentIdentifier: page.componentIdentifier,
462
555
  needsRefresh: needsRefresh(prevAssets, nextAssets)
463
556
  };
557
+ if (page.action !== "handleStreamResponse") {
558
+ meta.componentIdentifier = page.componentIdentifier;
559
+ }
560
+ return meta;
464
561
  }
465
562
  var MismatchedComponentError = class extends Error {
466
563
  constructor(message) {
@@ -468,10 +565,11 @@ var MismatchedComponentError = class extends Error {
468
565
  this.name = "MismatchedComponentError";
469
566
  }
470
567
  };
568
+ var defaultBeforeSave = (prevPage, receivedPage) => receivedPage;
471
569
  var remote = (path, {
472
570
  pageKey: targetPageKey,
473
571
  force = false,
474
- beforeSave = (prevPage, receivedPage) => receivedPage,
572
+ beforeSave = defaultBeforeSave,
475
573
  ...rest
476
574
  } = {}) => {
477
575
  targetPageKey = targetPageKey && urlToPageKey(targetPageKey);
@@ -490,10 +588,11 @@ var remote = (path, {
490
588
  pageKey = targetPageKey;
491
589
  }
492
590
  const meta = buildMeta(pageKey, json, superglue, rsp, fetchArgs);
493
- const existingId = pages[pageKey]?.componentIdentifier;
494
- const receivedId = json.componentIdentifier;
495
- if (!!existingId && existingId != receivedId && !force) {
496
- const message = `You cannot replace or update an existing page
591
+ if (json.action !== "handleStreamResponse") {
592
+ const existingId = pages[pageKey]?.componentIdentifier;
593
+ const receivedId = json.componentIdentifier;
594
+ if (!!existingId && existingId != receivedId && !force) {
595
+ const message = `You cannot replace or update an existing page
497
596
  located at pages["${currentPageKey}"] that has a componentIdentifier
498
597
  of "${existingId}" with the contents of a page response that has a
499
598
  componentIdentifier of "${receivedId}".
@@ -509,7 +608,8 @@ compatible with the page component associated with "${existingId}".
509
608
  Consider using data-sg-visit, the visit function, or redirect_back to
510
609
  the same page. Or if you're sure you want to proceed, use force: true.
511
610
  `;
512
- throw new MismatchedComponentError(message);
611
+ throw new MismatchedComponentError(message);
612
+ }
513
613
  }
514
614
  const page = beforeSave(pages[pageKey], json);
515
615
  return dispatch(saveAndProcessPage(pageKey, page)).then(() => meta);
@@ -553,11 +653,13 @@ var visit = (path, {
553
653
  const { superglue, pages = {} } = getState();
554
654
  const isGet = fetchArgs[1].method === "GET";
555
655
  const pageKey = calculatePageKey(rsp, isGet, currentPageKey);
556
- if (placeholderKey && hasPropsAt(path) && hasPlaceholder) {
557
- const existingId = pages[placeholderKey]?.componentIdentifier;
558
- const receivedId = json.componentIdentifier;
559
- if (!!existingId && existingId != receivedId) {
560
- const message = `You received a page response with a
656
+ const meta = buildMeta(pageKey, json, superglue, rsp, fetchArgs);
657
+ if (json.action !== "handleStreamResponse") {
658
+ if (placeholderKey && hasPropsAt(path) && hasPlaceholder) {
659
+ const existingId = pages[placeholderKey]?.componentIdentifier;
660
+ const receivedId = json.componentIdentifier;
661
+ if (!!existingId && existingId != receivedId) {
662
+ const message = `You received a page response with a
561
663
  componentIdentifier "${receivedId}" that is different than the
562
664
  componentIdentifier "${existingId}" located at ${placeholderKey}.
563
665
 
@@ -573,16 +675,17 @@ Check that you're rendering a page with a matching
573
675
  componentIdentifier, or consider using redirect_back_with_props_at
574
676
  to the same page.
575
677
  `;
576
- throw new MismatchedComponentError(message);
678
+ throw new MismatchedComponentError(message);
679
+ }
680
+ dispatch(copyPage({ from: placeholderKey, to: pageKey }));
577
681
  }
578
- dispatch(copyPage({ from: placeholderKey, to: pageKey }));
579
682
  }
580
- const meta = buildMeta(pageKey, json, superglue, rsp, fetchArgs);
581
683
  const visitMeta = {
582
684
  ...meta,
583
685
  navigationAction: calculateNavAction(
584
686
  meta,
585
687
  rsp,
688
+ json,
586
689
  isGet,
587
690
  pageKey,
588
691
  currentPageKey,
@@ -594,8 +697,11 @@ to the same page.
594
697
  }).catch((e) => handleFetchErr(e, fetchArgs, dispatch));
595
698
  };
596
699
  };
597
- function calculateNavAction(meta, rsp, isGet, pageKey, currentPageKey, revisit) {
700
+ function calculateNavAction(meta, rsp, json, isGet, pageKey, currentPageKey, revisit) {
598
701
  let navigationAction = "push";
702
+ if (json.action === "handleStreamResponse") {
703
+ return "none";
704
+ }
599
705
  if (!rsp.redirected && !isGet) {
600
706
  navigationAction = "replace";
601
707
  }
@@ -624,6 +730,155 @@ function calculatePageKey(rsp, isGet, currentPageKey) {
624
730
  return pageKey;
625
731
  }
626
732
 
733
+ // lib/action_creators/stream.ts
734
+ var streamPrepend = (fragments, data, options = {}) => {
735
+ return (dispatch) => {
736
+ if (options.saveAs) {
737
+ const { saveAs } = options;
738
+ dispatch(
739
+ saveFragment({
740
+ fragmentId: saveAs,
741
+ data
742
+ })
743
+ );
744
+ fragments.forEach((fragmentId) => {
745
+ dispatch(
746
+ prependToFragment({
747
+ fragmentId,
748
+ data: {
749
+ __id: saveAs
750
+ }
751
+ })
752
+ );
753
+ });
754
+ } else {
755
+ fragments.forEach((fragmentId) => {
756
+ dispatch(
757
+ prependToFragment({
758
+ fragmentId,
759
+ data
760
+ })
761
+ );
762
+ });
763
+ }
764
+ };
765
+ };
766
+ var streamAppend = (fragments, data, options = {}) => {
767
+ return (dispatch) => {
768
+ if (options.saveAs) {
769
+ const { saveAs } = options;
770
+ dispatch(
771
+ saveFragment({
772
+ fragmentId: saveAs,
773
+ data
774
+ })
775
+ );
776
+ fragments.forEach((fragmentId) => {
777
+ dispatch(
778
+ appendToFragment({
779
+ fragmentId,
780
+ data: {
781
+ __id: saveAs
782
+ }
783
+ })
784
+ );
785
+ });
786
+ } else {
787
+ fragments.forEach((fragmentId) => {
788
+ dispatch(
789
+ appendToFragment({
790
+ fragmentId,
791
+ data
792
+ })
793
+ );
794
+ });
795
+ }
796
+ };
797
+ };
798
+ var streamSave = (fragment, data) => {
799
+ return (dispatch) => {
800
+ dispatch(
801
+ saveFragment({
802
+ fragmentId: fragment,
803
+ data
804
+ })
805
+ );
806
+ };
807
+ };
808
+ var handleStreamMessage = (rawMessage) => {
809
+ return (dispatch) => {
810
+ const message = JSON.parse(rawMessage);
811
+ let nextMessage = message;
812
+ if (message.handler !== "refresh") {
813
+ message.fragments.reverse().forEach((fragment) => {
814
+ const { id, path } = fragment;
815
+ const node = getIn(nextMessage, path);
816
+ nextMessage = setIn(nextMessage, path, { __id: id });
817
+ dispatch(
818
+ saveFragment({
819
+ fragmentId: id,
820
+ data: node
821
+ })
822
+ );
823
+ });
824
+ }
825
+ if (nextMessage.action === "handleStreamMessage") {
826
+ if (nextMessage.handler === "append") {
827
+ dispatch(
828
+ streamAppend(
829
+ nextMessage.fragmentIds,
830
+ nextMessage.data,
831
+ nextMessage.options
832
+ )
833
+ );
834
+ }
835
+ if (nextMessage.handler === "prepend") {
836
+ dispatch(
837
+ streamPrepend(
838
+ nextMessage.fragmentIds,
839
+ nextMessage.data,
840
+ nextMessage.options
841
+ )
842
+ );
843
+ }
844
+ if (nextMessage.handler === "save") {
845
+ dispatch(streamSave(nextMessage.fragmentIds[0], nextMessage.data));
846
+ }
847
+ }
848
+ };
849
+ };
850
+ var handleStreamResponse = (response) => {
851
+ return (dispatch) => {
852
+ let nextResponse = response;
853
+ nextResponse.fragments.reverse().forEach((fragment) => {
854
+ const { id, path } = fragment;
855
+ const node = getIn(nextResponse, path);
856
+ nextResponse = setIn(nextResponse, path, { __id: id });
857
+ dispatch(
858
+ saveFragment({
859
+ fragmentId: id,
860
+ data: node
861
+ })
862
+ );
863
+ });
864
+ nextResponse.data.forEach((message) => {
865
+ if (message.handler === "append") {
866
+ dispatch(
867
+ streamAppend(message.fragmentIds, message.data, message.options)
868
+ );
869
+ }
870
+ if (message.handler === "prepend") {
871
+ dispatch(
872
+ streamPrepend(message.fragmentIds, message.data, message.options)
873
+ );
874
+ }
875
+ if (message.handler === "save") {
876
+ dispatch(streamSave(message.fragmentIds[0], message.data));
877
+ }
878
+ });
879
+ };
880
+ };
881
+
627
882
  // lib/action_creators/index.ts
628
883
  function fetchDeferments(pageKey, defers = []) {
629
884
  pageKey = urlToPageKey(pageKey);
@@ -658,57 +913,40 @@ function fetchDeferments(pageKey, defers = []) {
658
913
  };
659
914
  }
660
915
  function saveAndProcessPage(pageKey, page) {
661
- return (dispatch, getState) => {
916
+ return (dispatch) => {
662
917
  pageKey = urlToPageKey(pageKey);
663
- const { defers = [] } = page;
664
- if ("action" in page) {
665
- const prevPage = getState().pages[pageKey];
666
- dispatch(handleGraft({ pageKey, page }));
667
- const currentPage = getState().pages[pageKey];
668
- currentPage.fragments.forEach((fragment) => {
669
- const { type, path } = fragment;
670
- const currentFragment = getIn(currentPage, path);
671
- const prevFragment = getIn(prevPage, path);
672
- if (!prevFragment) {
673
- dispatch(
674
- updateFragments({
675
- name: type,
676
- pageKey,
677
- value: currentFragment,
678
- path
679
- })
680
- );
681
- } else if (currentFragment !== prevFragment) {
682
- dispatch(
683
- updateFragments({
684
- name: type,
685
- pageKey,
686
- value: currentFragment,
687
- previousValue: prevFragment,
688
- path
689
- })
690
- );
691
- }
692
- });
693
- } else {
694
- dispatch(saveResponse({ pageKey, page }));
695
- const currentPage = getState().pages[pageKey];
696
- currentPage.fragments.forEach((fragment) => {
697
- const { type, path } = fragment;
698
- const currentFragment = getIn(currentPage, path);
918
+ let nextPage = page;
919
+ page.fragments.reverse().forEach((fragment) => {
920
+ const { id, path } = fragment;
921
+ const node = getIn(nextPage, path);
922
+ nextPage = setIn(page, path, { __id: id });
923
+ dispatch(
924
+ saveFragment({
925
+ fragmentId: id,
926
+ data: node
927
+ })
928
+ );
929
+ });
930
+ if (nextPage.action === "graft") {
931
+ if (typeof nextPage.fragmentContext === "string") {
699
932
  dispatch(
700
- updateFragments({
701
- name: type,
702
- pageKey,
703
- value: currentFragment,
704
- path
933
+ handleFragmentGraft({
934
+ fragmentId: nextPage.fragmentContext,
935
+ response: nextPage
705
936
  })
706
937
  );
707
- });
938
+ } else {
939
+ dispatch(handleGraft({ pageKey, page: nextPage }));
940
+ }
941
+ } else if (nextPage.action === "handleStreamResponse") {
942
+ dispatch(handleStreamResponse(nextPage));
943
+ return Promise.resolve();
944
+ } else {
945
+ dispatch(saveResponse({ pageKey, page: nextPage }));
708
946
  }
709
947
  const hasFetch = typeof fetch != "undefined";
710
948
  if (hasFetch) {
711
- return dispatch(fetchDeferments(pageKey, defers)).then(
949
+ return dispatch(fetchDeferments(pageKey, nextPage.defers)).then(
712
950
  () => Promise.resolve()
713
951
  );
714
952
  } else {
@@ -718,18 +956,20 @@ function saveAndProcessPage(pageKey, page) {
718
956
  }
719
957
 
720
958
  export {
959
+ __commonJS,
960
+ __toESM,
721
961
  config,
722
962
  urlToPageKey,
723
963
  parsePageKey,
724
964
  argsForHistory,
725
965
  getIn,
726
966
  setIn,
967
+ lastRequestIds,
727
968
  ujsHandlers,
728
969
  GRAFTING_ERROR,
729
970
  GRAFTING_SUCCESS,
730
971
  saveResponse,
731
972
  handleGraft,
732
- updateFragments,
733
973
  copyPage,
734
974
  removePage,
735
975
  beforeFetch,
@@ -738,9 +978,17 @@ export {
738
978
  setCSRFToken,
739
979
  historyChange,
740
980
  setActivePage,
981
+ handleFragmentGraft,
982
+ saveFragment,
983
+ appendToFragment,
984
+ prependToFragment,
741
985
  MismatchedComponentError,
742
986
  remote,
743
987
  visit,
988
+ streamPrepend,
989
+ streamAppend,
990
+ streamSave,
991
+ handleStreamMessage,
744
992
  saveAndProcessPage
745
993
  };
746
- //# sourceMappingURL=chunk-7OMO5P27.mjs.map
994
+ //# sourceMappingURL=chunk-YS477OKK.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../lib/utils/url.ts","../lib/utils/helpers.ts","../lib/utils/immutability.ts","../lib/config.ts","../lib/utils/limited_set.ts","../lib/utils/request.ts","../lib/utils/ujs.ts","../lib/utils/window.ts","../lib/actions.ts","../lib/action_creators/requests.ts","../lib/action_creators/stream.ts","../lib/action_creators/index.ts"],"sourcesContent":["import { PageKey } from '../types'\n\nconst FAKE_ORIGIN = 'https://example.com'\n\nexport function pathQuery(url: string): string {\n const { pathname, search: query } = new URL(url, FAKE_ORIGIN)\n\n return pathname + query\n}\n\nexport function pathQueryHash(url: string): string {\n const { pathname, hash, search: query } = new URL(url, FAKE_ORIGIN)\n\n return pathname + query + hash\n}\n\nexport function hasPropsAt(url: string): boolean {\n const { searchParams } = new URL(url, FAKE_ORIGIN)\n\n return searchParams.has('props_at')\n}\n\nexport function propsAtParam(url: string): string | null {\n const { searchParams } = new URL(url, FAKE_ORIGIN)\n\n return searchParams.get('props_at')\n}\n\nexport function withFormatJson(url: string): string {\n const parsed = new URL(url, FAKE_ORIGIN)\n parsed.searchParams.set('format', 'json')\n\n return parsed.href.replace(parsed.origin, '')\n}\n\nexport function removePropsAt(url: string): string {\n const parsed = new URL(url, FAKE_ORIGIN)\n parsed.searchParams.delete('props_at')\n\n return parsed.href.replace(parsed.origin, '')\n}\n\n/**\n * Converts a url to a PageKey.\n *\n * @param url\n * @returns\n */\nexport function urlToPageKey(url: string): PageKey {\n const parsed = new URL(url, FAKE_ORIGIN)\n parsed.searchParams.delete('props_at')\n parsed.searchParams.delete('format')\n\n return pathQuery(parsed.toString())\n}\n\nexport function withoutHash(url: string): string {\n const parsed = new URL(url, FAKE_ORIGIN)\n parsed.hash = ''\n\n return parsed.href.replace(parsed.origin, '')\n}\n\nexport function formatForXHR(url: string): string {\n const formats = [withoutHash, withFormatJson]\n\n return formats.reduce((memo, f) => f(memo), url)\n}\n\nexport function parsePageKey(pageKey: PageKey) {\n const { pathname, searchParams } = new URL(pageKey, FAKE_ORIGIN)\n\n const search = Object.fromEntries(searchParams)\n\n return {\n pathname,\n search,\n }\n}\n","import { GraftResponse, HistoryState, SaveResponse } from '../types'\nimport { urlToPageKey } from './url'\n\nexport function isGraft(page: GraftResponse | SaveResponse): boolean {\n return 'action' in page && page.action === 'graft'\n}\n\nexport function extractNodeAndPath(page: GraftResponse): {\n node: unknown\n pathToNode: string\n} {\n const { data: node, action, path: pathToNode } = page\n\n if (action === 'graft') {\n return { node, pathToNode }\n } else {\n const errMsg =\n 'Expected page to be a graft response rendered from node filtering.'\n throw new Error(errMsg)\n }\n}\n\nexport function argsForHistory(path: string): [string, HistoryState] {\n const pageKey = urlToPageKey(path)\n\n return [\n path,\n {\n superglue: true,\n pageKey,\n posX: 0,\n posY: 0,\n },\n ]\n}\n","// These were taken from Scour.js\n// Then, modified to respect the id=0 keypath\n\nimport { JSONMappable, JSONValue, Keypath } from '../types'\n\nconst canLookAhead = /^[\\da-zA-Z\\-_]+=[\\da-zA-Z\\-_]+$/\n\nclass KeyPathError extends Error {\n constructor(message: string) {\n super(message)\n this.name = 'KeyPathError'\n }\n}\n\n/**\n * Retrieves data from a JSON object using a {@link Keypath}\n *\n * @param node\n * @param path\n * @returns\n */\nfunction getIn(node: JSONMappable, path: Keypath): JSONValue {\n const keyPath = normalizeKeyPath(path)\n let result: JSONValue = node\n let i: number\n\n for (i = 0; i < keyPath.length; i++) {\n const key = keyPath[i]\n\n if (typeof result === 'object' && result !== null) {\n if (!Array.isArray(result) && canLookAhead.test(key)) {\n throw new KeyPathError(\n `Expected to find an Array when using the key: ${key}`\n )\n }\n\n result = atKey(result, key)\n } else {\n throw new KeyPathError(\n `Expected to traverse an Array or Obj, got ${JSON.stringify(result)}`\n )\n }\n }\n\n if (i === keyPath.length) {\n return result\n } else {\n return undefined\n }\n}\n\nfunction clone(node: JSONMappable): JSONMappable {\n return Array.isArray(node) ? [].slice.call(node) : { ...node }\n}\n\nfunction getKey(node: JSONMappable, key: string): string | number | never {\n if (Array.isArray(node) && Number.isNaN(Number(key))) {\n const key_parts = Array.from(key.split('='))\n const attr = key_parts[0]\n const id = key_parts[1]\n\n if (!id || !attr) {\n return key\n }\n\n let i: number\n let child: JSONValue\n\n for (i = 0; i < node.length; i++) {\n child = node[i]\n if (\n typeof child === 'object' &&\n !Array.isArray(child) &&\n child !== null\n ) {\n const val = child[attr]\n if (val && val.toString() === id) {\n break\n }\n } else {\n throw new KeyPathError(`Could not look ahead ${key} at ${child}`)\n }\n }\n\n if (i === node.length) {\n throw new KeyPathError(`Could not find ${key} while looking ahead`)\n }\n\n return i\n } else {\n return key\n }\n}\n\nfunction atKey(node: JSONMappable, key: string) {\n const actualKey = getKey(node, key)\n\n if (Array.isArray(node)) {\n return node[actualKey as number]\n } else {\n return node[actualKey]\n }\n}\n\nfunction normalizeKeyPath(path: string): string[] {\n if (typeof path === 'string') {\n path = path.replace(/ /g, '')\n if (path === '') {\n return []\n }\n\n return path.split('.')\n } else {\n return path\n }\n}\n/**\n * Sets data into a JSON object using a {@link Keypath}\n *\n * @param object\n * @param path\n * @param value\n * @returns\n */\nfunction setIn<T extends JSONMappable>(\n object: T,\n path: string,\n value: JSONValue\n): T | never {\n const keypath = normalizeKeyPath(path)\n\n const results: {\n 0: T\n [key: number]: JSONValue\n } = { 0: object }\n\n const parents: {\n 0: T\n [key: number]: JSONValue\n } = { 0: object }\n\n let i: number\n\n for (i = 0; i < keypath.length; i++) {\n const parent = parents[i]\n\n if (!(typeof parent === 'object' && parent !== null)) {\n throw new KeyPathError(\n `Expected to traverse an Array or Obj, got ${JSON.stringify(parent)}`\n )\n }\n\n const child = atKey(parent, keypath[i])\n parents[i + 1] = child\n }\n\n results[keypath.length] = value\n\n for (i = keypath.length - 1; i >= 0; i--) {\n // Parents will always have a JSONValue at\n // keypath.length so this loop skips that one element\n // Every other element in parents is a JSONMappable\n const target = clone(parents[i] as JSONMappable)\n results[i] = target\n const key = getKey(results[i] as JSONMappable, keypath[i])\n if (Array.isArray(target)) {\n target[key as number] = results[i + 1]\n } else {\n target[key] = results[i + 1]\n }\n }\n\n return results[0]\n}\n\nexport { getIn, setIn, KeyPathError }\n","export const config = {\n baseUrl: '',\n maxPages: 20,\n}\n","export class LimitedSet extends Set {\n private maxSize: number\n\n constructor(maxSize: number) {\n super()\n this.maxSize = maxSize\n }\n\n add(value: unknown) {\n if (this.size >= this.maxSize) {\n const iterator = this.values()\n const oldestValue = iterator.next().value\n this.delete(oldestValue)\n }\n super.add(value)\n\n return this\n }\n}\n","import { formatForXHR } from './url'\nimport { config } from '../config'\nimport { BasicRequestInit, ParsedResponse, RootState } from '../types'\nimport { LimitedSet } from './limited_set'\nimport { v4 as uuidv4 } from 'uuid'\n\nexport const lastRequestIds = new LimitedSet(20)\n\nexport function isValidResponse(xhr: Response): boolean {\n return isValidContent(xhr) && !downloadingFile(xhr)\n}\n\nexport function isValidContent(rsp: Response): boolean {\n const contentType = rsp.headers.get('content-type')\n const jsContent = /^(?:application\\/json)(?:;|$)/\n\n return !!(contentType && contentType.match(jsContent))\n}\n\nfunction downloadingFile(xhr: Response): boolean {\n const disposition = xhr.headers.get('content-disposition')\n\n return !!(disposition && disposition.match(/^attachment/) !== null)\n}\n\nclass SuperglueResponseError extends Error {\n response: Response\n\n constructor(message: string) {\n super(message)\n this.name = 'SuperglueResponseError'\n }\n}\n\nexport function validateResponse(args: ParsedResponse): ParsedResponse {\n const { rsp } = args\n if (isValidResponse(rsp)) {\n return args\n } else {\n const error = new SuperglueResponseError('Invalid Superglue Response')\n error.response = rsp\n throw error\n }\n}\n\nexport function handleServerErrors(args: ParsedResponse): ParsedResponse {\n const { rsp } = args\n if (!rsp.ok && rsp.status !== 422) {\n if (rsp.status === 406) {\n console.error(\n \"Superglue encountered a 406 Not Acceptable response. This can happen if you used respond_to and didn't specify format.json in the block. Try adding it to your respond_to. For example:\\n\\n\" +\n 'respond_to do |format|\\n' +\n ' format.html\\n' +\n ' format.json\\n' +\n ' format.csv\\n' +\n 'end'\n )\n }\n const error = new SuperglueResponseError(rsp.statusText)\n error.response = rsp\n throw error\n }\n return args\n}\n\nexport function argsForFetch(\n getState: () => RootState,\n pathQuery: string,\n {\n method = 'GET',\n headers = {},\n body = '',\n signal,\n ...rest\n }: BasicRequestInit = {}\n): [string, BasicRequestInit] {\n method = method.toUpperCase()\n const currentState = getState().superglue\n\n const nextHeaders = { ...headers }\n nextHeaders['x-requested-with'] = 'XMLHttpRequest'\n nextHeaders['accept'] = 'application/json'\n nextHeaders['x-superglue-request'] = 'true'\n\n const requestId = uuidv4()\n lastRequestIds.add(requestId)\n nextHeaders['X-Superglue-Request-Id'] = requestId\n\n if (method != 'GET' && method != 'HEAD') {\n nextHeaders['content-type'] = 'application/json'\n }\n\n if (body instanceof FormData) {\n delete nextHeaders['content-type']\n }\n\n if (currentState.csrfToken) {\n nextHeaders['x-csrf-token'] = currentState.csrfToken\n }\n\n const fetchPath = new URL(formatForXHR(pathQuery), config.baseUrl)\n\n const credentials = 'same-origin'\n\n if (!(method == 'GET' || method == 'HEAD')) {\n nextHeaders['x-http-method-override'] = method\n method = 'POST'\n }\n\n const options: BasicRequestInit = {\n method,\n headers: nextHeaders,\n body,\n credentials,\n signal,\n }\n\n if (currentState.currentPageKey) {\n const referrer = new URL(currentState.currentPageKey, config.baseUrl)\n\n options.referrer = referrer.toString()\n }\n\n if (method == 'GET' || method == 'HEAD') {\n if (options.body instanceof FormData) {\n const allData = new URLSearchParams(\n options.body as unknown as Record<string, string>\n )\n\n // Form data should override anything in the URL params First we\n // delete every key. Then append the new keys accounting for\n // duplicate keys that represent structural arrays.\n allData.forEach((value, key) => fetchPath.searchParams.delete(key))\n allData.forEach((value, key) => fetchPath.searchParams.append(key, value))\n }\n\n delete options.body\n }\n\n return [fetchPath.toString(), { ...options, ...rest }]\n}\n\nexport function extractJSON(rsp: Response): PromiseLike<ParsedResponse> {\n return rsp\n .json()\n .then((json) => {\n return { rsp, json }\n })\n .catch((e) => {\n e.response = rsp\n throw e\n })\n}\n\nexport function parseResponse(prm: Response): PromiseLike<ParsedResponse> {\n return Promise.resolve(prm)\n .then(extractJSON)\n .then(handleServerErrors)\n .then(validateResponse)\n}\n","import {\n Handlers,\n UJSHandlers,\n SuperglueStore,\n ApplicationVisit,\n ApplicationRemote,\n RemoteProps,\n VisitProps,\n} from '../types'\n\nexport class HandlerBuilder {\n public attributePrefix: string\n public visit: ApplicationVisit\n public remote: ApplicationRemote\n private store: SuperglueStore\n\n constructor({\n ujsAttributePrefix,\n visit,\n remote,\n store,\n }: {\n ujsAttributePrefix: string\n visit: ApplicationVisit\n remote: ApplicationRemote\n store: SuperglueStore\n }) {\n this.attributePrefix = ujsAttributePrefix\n this.isUJS = this.isUJS.bind(this)\n this.store = store\n\n this.handleSubmit = this.handleSubmit.bind(this)\n this.handleClick = this.handleClick.bind(this)\n\n this.visit = visit\n this.remote = remote\n this.visitOrRemote = this.visitOrRemote.bind(this)\n }\n\n retrieveLink(target: Element): HTMLAnchorElement | undefined {\n const link = target.closest<HTMLAnchorElement>('a')\n if (link && link.href.length !== 0) {\n return link\n }\n }\n\n isNonStandardClick(\n event: React.MouseEvent<HTMLDivElement, MouseEvent>\n ): boolean {\n return (\n event.button > 0 ||\n event.metaKey ||\n event.ctrlKey ||\n event.shiftKey ||\n event.altKey\n )\n }\n\n isUJS(node: HTMLFormElement | HTMLAnchorElement): boolean {\n const hasVisit = !!node.getAttribute(this.attributePrefix + '-visit')\n const hasRemote = !!node.getAttribute(this.attributePrefix + '-remote')\n\n return hasVisit || hasRemote\n }\n\n handleSubmit(event: React.FormEvent<HTMLDivElement>): void {\n const form = event.target\n\n if (!(form instanceof HTMLFormElement)) {\n return\n }\n\n if (!this.isUJS(form)) {\n return\n }\n\n event.preventDefault()\n\n const url = form.getAttribute('action')\n if (!url) {\n return\n }\n\n const method = (form.getAttribute('method') || 'POST').toUpperCase()\n\n this.visitOrRemote(form, url, {\n method,\n body: new FormData(form),\n })\n }\n\n handleClick(event: React.MouseEvent<HTMLDivElement, MouseEvent>): void {\n if (!(event.target instanceof Element)) {\n return\n }\n\n const link = this.retrieveLink(event.target)\n const isNonStandard = this.isNonStandardClick(event)\n if (!link || isNonStandard || !this.isUJS(link)) {\n return\n }\n\n event.preventDefault()\n const url = link.getAttribute('href')\n if (!url) {\n return\n }\n\n this.visitOrRemote(link, url, { method: 'GET' })\n }\n\n visitOrRemote(\n linkOrForm: HTMLAnchorElement | HTMLFormElement,\n url: string,\n opts: RemoteProps | VisitProps\n ): void {\n const dataset = { ...linkOrForm.dataset }\n\n if (linkOrForm.getAttribute(this.attributePrefix + '-visit')) {\n this.visit(url, { ...opts, dataset })\n }\n\n if (linkOrForm.getAttribute(this.attributePrefix + '-remote')) {\n const { currentPageKey } = this.store.getState().superglue\n this.remote(url, {\n ...opts,\n pageKey: currentPageKey,\n dataset,\n })\n }\n }\n\n handlers(): Handlers {\n return {\n onClick: this.handleClick,\n onSubmit: this.handleSubmit,\n }\n }\n}\n\nexport const ujsHandlers: UJSHandlers = ({\n ujsAttributePrefix,\n visit,\n remote,\n store,\n}) => {\n const builder = new HandlerBuilder({\n visit,\n remote,\n ujsAttributePrefix,\n store,\n })\n\n return builder.handlers()\n}\n","export function needsRefresh(\n prevAssets: string[],\n newAssets: string[]\n): boolean {\n if (prevAssets && newAssets) {\n const hasNewAssets = !newAssets.every((asset) => prevAssets.includes(asset))\n return hasNewAssets\n } else {\n return false\n }\n}\n","import { createAction } from '@reduxjs/toolkit'\nimport {\n FetchArgs,\n PageKey,\n GraftResponse,\n SaveResponse,\n JSONMappable,\n} from './types'\nimport { urlToPageKey } from './utils'\n\nexport const GRAFTING_ERROR = '@@superglue/GRAFTING_ERROR'\nexport const GRAFTING_SUCCESS = '@@superglue/GRAFTING_SUCCESS'\n\nexport const saveResponse = createAction(\n '@@superglue/SAVE_RESPONSE',\n ({ pageKey, page }: { pageKey: string; page: SaveResponse }) => {\n pageKey = urlToPageKey(pageKey)\n\n return {\n payload: {\n pageKey,\n page,\n },\n }\n }\n)\n\nexport const handleGraft = createAction(\n '@@superglue/HANDLE_GRAFT',\n ({ pageKey, page }: { pageKey: string; page: GraftResponse }) => {\n pageKey = urlToPageKey(pageKey)\n\n return {\n payload: {\n page,\n pageKey,\n },\n }\n }\n)\n\nexport const superglueError = createAction<{ message: string }>(\n '@@superglue/ERROR'\n)\n\n/**\n * A redux action you can dispatch to copy a page from one pageKey to another. Its\n * a very useful way to create optimistic updates with a URL change. For example:\n *\n * ```\n * import { copyPage, remote } from '@thoughtbot/superglue'\n *\n * dispatch(copyPage({ from: originalKey, to: targetKey}))\n *\n * ... make edits to target page and finally\n *\n * navigateTo(targetKey)\n * ```\n */\nexport const copyPage = createAction<{ from: PageKey; to: PageKey }>(\n '@@superglue/COPY_PAGE'\n)\n\n/**\n * A redux action you can dispatch to remove a page from your store.\n *\n * ```\n * import { removePage } from '@thoughtbot/superglue'\n *\n * dispatch(removePage({ pageKey: '/delete_me_please\"}))\n * ```\n */\nexport const removePage = createAction<{ pageKey: PageKey }>(\n '@@superglue/REMOVE_PAGE'\n)\n\n/**\n * A redux action called before a `fetch` takes place. It will fire in `remote`\n * and `visit`. You can hook into this event in your redux slices like this:\n *\n * ```\n * import { beforeFetch } from '@thoughtbot/superglue'\n *\n * export const exampleSlice = createSlice({\n * name: 'Example',\n * initialState: {},\n * extraReducers: (builder) => {\n * builder.addCase(beforeFetch, (state, action) => {\n * ```\n */\nexport const beforeFetch = createAction<{ fetchArgs: FetchArgs }>(\n '@@superglue/BEFORE_FETCH'\n)\n\n/**\n * A redux action called before a `visit` takes place. You can hook into this event\n * in your redux slices like this:\n *\n * ```\n * import { beforeVisit } from '@thoughtbot/superglue'\n *\n * export const exampleSlice = createSlice({\n * name: 'Example',\n * initialState: {},\n * extraReducers: (builder) => {\n * builder.addCase(beforeVisit, (state, action) => {\n * ```\n */\nexport const beforeVisit = createAction<{\n currentPageKey: PageKey\n fetchArgs: FetchArgs\n}>('@@superglue/BEFORE_VISIT')\n\n/**\n * A redux action called before `remote` takes place. You can hook into this event\n * in your redux slices like this:\n *\n * ```\n * import { beforeRemote } from '@thoughtbot/superglue'\n *\n * export const exampleSlice = createSlice({\n * name: 'Example',\n * initialState: {},\n * extraReducers: (builder) => {\n * builder.addCase(beforeRemote, (state, action) => {\n * ```\n */\nexport const beforeRemote = createAction<{\n currentPageKey: PageKey\n fetchArgs: FetchArgs\n}>('@@superglue/BEFORE_REMOTE')\n\nexport const setCSRFToken = createAction<{\n csrfToken: string | undefined\n}>('@@superglue/SET_CSRF_TOKEN')\n\nexport const historyChange = createAction<{\n pageKey: PageKey\n}>('@@superglue/HISTORY_CHANGE')\n\nexport const setActivePage = createAction<{\n pageKey: PageKey\n}>('@@superglue/SET_ACTIVE_PAGE')\n\nexport const handleFragmentGraft = createAction(\n '@@superglue/HANDLE_FRAGMENT_GRAFT',\n ({\n fragmentId,\n response,\n }: {\n fragmentId: string\n response: GraftResponse\n }) => {\n return {\n payload: {\n response,\n fragmentId,\n },\n }\n }\n)\n\nexport const saveFragment = createAction(\n '@@superglue/SAVE_FRAGMENT',\n ({ fragmentId, data }: { fragmentId: string; data: JSONMappable }) => {\n return {\n payload: {\n fragmentId,\n data,\n },\n }\n }\n)\n\nexport const appendToFragment = createAction(\n '@@superglue/APPEND_TO_FRAGMENT',\n ({ data, fragmentId }: { data: JSONMappable; fragmentId: string }) => {\n return {\n payload: {\n data,\n fragmentId,\n },\n }\n }\n)\n\nexport const prependToFragment = createAction(\n '@@superglue/PREPEND_TO_FRAGMENT',\n ({ data, fragmentId }: { data: JSONMappable; fragmentId: string }) => {\n return {\n payload: {\n data,\n fragmentId: fragmentId,\n },\n }\n }\n)\n","import {\n argsForFetch,\n parseResponse,\n needsRefresh,\n urlToPageKey,\n hasPropsAt,\n propsAtParam,\n removePropsAt,\n} from '../utils'\nimport {\n beforeFetch,\n beforeVisit,\n beforeRemote,\n copyPage,\n superglueError,\n} from '../actions'\nimport { saveAndProcessPage } from './index'\nimport {\n FetchArgs,\n PageResponse,\n Page,\n SuperglueState,\n Meta,\n Dispatch,\n RemoteCreator,\n VisitCreator,\n NavigationAction,\n VisitMeta,\n BeforeSave,\n} from '../types'\n\nfunction handleFetchErr(\n err: Error,\n fetchArgs: FetchArgs,\n dispatch: Dispatch\n): never {\n dispatch(superglueError({ message: err.message }))\n throw err\n}\n\nfunction buildMeta(\n pageKey: string,\n page: PageResponse,\n state: SuperglueState,\n rsp: Response,\n fetchArgs: FetchArgs\n): Meta {\n const { assets: prevAssets } = state\n const { assets: nextAssets } = page\n\n const meta: Meta = {\n pageKey,\n page,\n redirected: rsp.redirected,\n rsp,\n fetchArgs,\n needsRefresh: needsRefresh(prevAssets, nextAssets),\n }\n\n if (page.action !== 'handleStreamResponse') {\n meta.componentIdentifier = page.componentIdentifier\n }\n\n return meta\n}\n\nexport class MismatchedComponentError extends Error {\n constructor(message: string) {\n super(message)\n this.name = 'MismatchedComponentError'\n }\n}\n\nconst defaultBeforeSave: BeforeSave = (prevPage, receivedPage) => receivedPage\n\nexport const remote: RemoteCreator = (\n path,\n {\n pageKey: targetPageKey,\n force = false,\n beforeSave = defaultBeforeSave,\n ...rest\n } = {}\n) => {\n targetPageKey = targetPageKey && urlToPageKey(targetPageKey)\n\n return (dispatch, getState) => {\n const fetchArgs = argsForFetch(getState, path, rest)\n const currentPageKey = getState().superglue.currentPageKey\n\n dispatch(beforeRemote({ currentPageKey, fetchArgs }))\n dispatch(beforeFetch({ fetchArgs }))\n\n return fetch(...fetchArgs)\n .then(parseResponse)\n .then(({ rsp, json }) => {\n const { superglue, pages = {} } = getState()\n\n let pageKey\n if (targetPageKey === undefined) {\n const isGet = fetchArgs[1].method === 'GET'\n pageKey = calculatePageKey(rsp, isGet, currentPageKey)\n } else {\n pageKey = targetPageKey\n }\n\n const meta = buildMeta(pageKey, json, superglue, rsp, fetchArgs)\n\n if (json.action !== 'handleStreamResponse') {\n const existingId = pages[pageKey]?.componentIdentifier\n const receivedId = json.componentIdentifier\n if (!!existingId && existingId != receivedId && !force) {\n const message = `You cannot replace or update an existing page\nlocated at pages[\"${currentPageKey}\"] that has a componentIdentifier\nof \"${existingId}\" with the contents of a page response that has a\ncomponentIdentifier of \"${receivedId}\".\n\nThis can happen if you're using data-sg-remote or remote but your\nresponse redirected to a page with a different componentIdentifier\nthan the target page. \n\nThis limitation exists because the resulting page shape from grafting\n\"${receivedId}\"'s \"${propsAtParam(path)}\" into \"${existingId}\" may not be\ncompatible with the page component associated with \"${existingId}\".\n\nConsider using data-sg-visit, the visit function, or redirect_back to\nthe same page. Or if you're sure you want to proceed, use force: true.\n `\n throw new MismatchedComponentError(message)\n }\n }\n\n const page = beforeSave(pages[pageKey], json)\n return dispatch(saveAndProcessPage(pageKey, page)).then(() => meta)\n })\n .catch((e) => handleFetchErr(e, fetchArgs, dispatch))\n }\n}\n\nlet lastVisitController = {\n /* eslint-disable-next-line @typescript-eslint/no-unused-vars */\n abort: (_reason: string) => {\n // noop\n },\n}\n\nexport const visit: VisitCreator = (\n path,\n {\n placeholderKey,\n beforeSave = (prevPage: Page, receivedPage: PageResponse) => receivedPage,\n revisit = false,\n ...rest\n } = {}\n) => {\n return (dispatch, getState) => {\n const currentPageKey = getState().superglue.currentPageKey\n placeholderKey =\n (placeholderKey && urlToPageKey(placeholderKey)) || currentPageKey\n const hasPlaceholder = placeholderKey in getState().pages\n\n if (hasPropsAt(path) && !hasPlaceholder) {\n console.warn(\n `Could not find placeholder with key ${placeholderKey} in state. The props_at param will be ignored`\n )\n path = removePropsAt(path)\n }\n\n const controller = new AbortController()\n const { signal } = controller\n const fetchArgs = argsForFetch(getState, path, {\n ...rest,\n signal,\n })\n\n dispatch(beforeVisit({ currentPageKey, fetchArgs }))\n dispatch(beforeFetch({ fetchArgs }))\n\n lastVisitController.abort(\n 'Aborting the previous `visit`. There can be one visit at a time. Use `remote` if there is a need for async requests.'\n )\n lastVisitController = controller\n\n return fetch(...fetchArgs)\n .then(parseResponse)\n .then(({ rsp, json }) => {\n const { superglue, pages = {} } = getState()\n const isGet = fetchArgs[1].method === 'GET'\n const pageKey = calculatePageKey(rsp, isGet, currentPageKey)\n\n const meta = buildMeta(pageKey, json, superglue, rsp, fetchArgs)\n\n if (json.action !== 'handleStreamResponse') {\n if (placeholderKey && hasPropsAt(path) && hasPlaceholder) {\n const existingId = pages[placeholderKey]?.componentIdentifier\n const receivedId = json.componentIdentifier\n if (!!existingId && existingId != receivedId) {\n const message = `You received a page response with a\ncomponentIdentifier \"${receivedId}\" that is different than the\ncomponentIdentifier \"${existingId}\" located at ${placeholderKey}.\n\nThis can happen if you're using data-sg-visit or visit with a\nprops_at param, but the response redirected to a page with a\ndifferent componentIdentifier than the target page. \n\nThis limitation exists because the resulting page shape from grafting\n\"${receivedId}\"'s \"${propsAtParam(path)}\" into \"${existingId}\" may not be\ncompatible with the page component associated with \"${existingId}\".\n\nCheck that you're rendering a page with a matching\ncomponentIdentifier, or consider using redirect_back_with_props_at\nto the same page.\n `\n throw new MismatchedComponentError(message)\n }\n dispatch(copyPage({ from: placeholderKey, to: pageKey }))\n }\n }\n const visitMeta: VisitMeta = {\n ...meta,\n navigationAction: calculateNavAction(\n meta,\n rsp,\n json,\n isGet,\n pageKey,\n currentPageKey,\n revisit\n ),\n }\n\n const page = beforeSave(pages[pageKey], json)\n return dispatch(saveAndProcessPage(pageKey, page)).then(() => visitMeta)\n })\n .catch((e) => handleFetchErr(e, fetchArgs, dispatch))\n }\n}\n\nfunction calculateNavAction(\n meta: Meta,\n rsp: Response,\n json: PageResponse,\n isGet: boolean,\n pageKey: string,\n currentPageKey: string,\n revisit: boolean\n) {\n let navigationAction: NavigationAction = 'push'\n\n if (json.action === 'handleStreamResponse') {\n return 'none'\n }\n\n if (!rsp.redirected && !isGet) {\n navigationAction = 'replace'\n }\n const isSamePage = pageKey == currentPageKey\n if (isSamePage) {\n navigationAction = 'none'\n }\n if (revisit && isGet) {\n if (rsp.redirected) {\n navigationAction = 'replace'\n } else {\n navigationAction = 'none'\n }\n }\n\n return navigationAction\n}\n\nfunction calculatePageKey(\n rsp: Response,\n isGet: boolean,\n currentPageKey: string\n) {\n let pageKey = urlToPageKey(rsp.url)\n if (!isGet && !rsp.redirected) {\n pageKey = currentPageKey\n }\n\n const contentLocation = rsp.headers.get('content-location')\n if (contentLocation) {\n pageKey = urlToPageKey(contentLocation)\n }\n return pageKey\n}\n","import { ThunkAction } from 'redux-thunk'\nimport { Action } from '@reduxjs/toolkit'\nimport { setIn, getIn } from '../utils'\nimport { appendToFragment, prependToFragment, saveFragment } from '../actions'\nimport { JSONMappable, RootState, StreamResponse } from '../types'\nimport { StreamMessage } from '../hooks/useStreamSource'\n\nexport type StreamThunk = ThunkAction<void, RootState, undefined, Action>\nexport type StreamHandleThunk = ThunkAction<void, RootState, undefined, Action>\n\nexport interface StreamThunkOptions {\n saveAs?: string\n}\n\n/**\n * Stream thunk equivalent to StreamActions.prepend()\n * Prepends data to specified fragments, optionally saving as a new fragment\n */\nexport const streamPrepend = (\n fragments: string[],\n data: JSONMappable,\n options: StreamThunkOptions = {}\n): StreamThunk => {\n return (dispatch) => {\n if (options.saveAs) {\n const { saveAs } = options\n dispatch(\n saveFragment({\n fragmentId: saveAs,\n data,\n })\n )\n\n fragments.forEach((fragmentId) => {\n dispatch(\n prependToFragment({\n fragmentId,\n data: {\n __id: saveAs,\n },\n })\n )\n })\n } else {\n fragments.forEach((fragmentId) => {\n dispatch(\n prependToFragment({\n fragmentId: fragmentId,\n data: data,\n })\n )\n })\n }\n }\n}\n\n/**\n * Stream thunk equivalent to StreamActions.append()\n * Appends data to specified fragments, optionally saving as a new fragment\n */\nexport const streamAppend = (\n fragments: string[],\n data: JSONMappable,\n options: StreamThunkOptions = {}\n): StreamThunk => {\n return (dispatch) => {\n if (options.saveAs) {\n const { saveAs } = options\n dispatch(\n saveFragment({\n fragmentId: saveAs,\n data,\n })\n )\n\n fragments.forEach((fragmentId) => {\n dispatch(\n appendToFragment({\n fragmentId,\n data: {\n __id: saveAs,\n },\n })\n )\n })\n } else {\n fragments.forEach((fragmentId) => {\n dispatch(\n appendToFragment({\n fragmentId,\n data,\n })\n )\n })\n }\n }\n}\n\n/**\n * Stream thunk equivalent to StreamActions.save()\n * Saves data to a specific fragment\n */\nexport const streamSave = (\n fragment: string,\n data: JSONMappable\n): StreamThunk => {\n return (dispatch) => {\n dispatch(\n saveFragment({\n fragmentId: fragment,\n data,\n })\n )\n }\n}\n\nexport const handleStreamMessage = (rawMessage: string): StreamHandleThunk => {\n return (dispatch) => {\n const message = JSON.parse(rawMessage) as StreamMessage\n\n let nextMessage = message\n\n if (message.handler !== 'refresh') {\n message.fragments.reverse().forEach((fragment) => {\n const { id, path } = fragment\n const node = getIn(nextMessage as JSONMappable, path) as JSONMappable\n nextMessage = setIn(nextMessage, path, { __id: id })\n\n dispatch(\n saveFragment({\n fragmentId: id,\n data: node,\n })\n )\n })\n }\n\n if (nextMessage.action === 'handleStreamMessage') {\n if (nextMessage.handler === 'append') {\n dispatch(\n streamAppend(\n nextMessage.fragmentIds,\n nextMessage.data,\n nextMessage.options\n )\n )\n }\n\n if (nextMessage.handler === 'prepend') {\n dispatch(\n streamPrepend(\n nextMessage.fragmentIds,\n nextMessage.data,\n nextMessage.options\n )\n )\n }\n\n if (nextMessage.handler === 'save') {\n dispatch(streamSave(nextMessage.fragmentIds[0], nextMessage.data))\n }\n }\n }\n}\n\nexport const handleStreamResponse = (\n response: StreamResponse\n): StreamHandleThunk => {\n return (dispatch) => {\n let nextResponse = response\n\n nextResponse.fragments.reverse().forEach((fragment) => {\n const { id, path } = fragment\n const node = getIn(nextResponse as JSONMappable, path) as JSONMappable\n nextResponse = setIn(nextResponse, path, { __id: id })\n\n dispatch(\n saveFragment({\n fragmentId: id,\n data: node,\n })\n )\n })\n\n nextResponse.data.forEach((message) => {\n if (message.handler === 'append') {\n dispatch(\n streamAppend(message.fragmentIds, message.data, message.options)\n )\n }\n\n if (message.handler === 'prepend') {\n dispatch(\n streamPrepend(message.fragmentIds, message.data, message.options)\n )\n }\n\n if (message.handler === 'save') {\n dispatch(streamSave(message.fragmentIds[0], message.data))\n }\n })\n }\n}\n","import { urlToPageKey, getIn, setIn, propsAtParam } from '../utils'\nimport {\n saveResponse,\n GRAFTING_ERROR,\n GRAFTING_SUCCESS,\n handleGraft,\n saveFragment,\n handleFragmentGraft,\n} from '../actions'\nimport { remote } from './requests'\nimport {\n SaveAndProcessPageThunk,\n DefermentThunk,\n Defer,\n JSONMappable,\n PageResponse,\n} from '../types'\nimport { handleStreamResponse } from './stream'\nexport * from './requests'\n\nfunction fetchDeferments(\n pageKey: string,\n defers: Defer[] = []\n): DefermentThunk {\n pageKey = urlToPageKey(pageKey)\n return (dispatch) => {\n const fetches = defers\n .filter(({ type }) => type === 'auto')\n .map(function ({\n url,\n successAction = GRAFTING_SUCCESS,\n failAction = GRAFTING_ERROR,\n }) {\n // props_at will always be present in a graft response\n // That's why this is marked `as string`\n const keyPath = propsAtParam(url) as string\n\n return dispatch(remote(url, { pageKey }))\n .then(() => {\n dispatch({\n type: successAction,\n payload: {\n pageKey,\n keyPath,\n },\n })\n })\n .catch((err) => {\n dispatch({\n type: failAction,\n payload: {\n url,\n err,\n pageKey,\n keyPath,\n },\n })\n })\n })\n\n return Promise.all(fetches)\n }\n}\n\n/**\n * Save and process a rendered view from PropsTemplate. This is the primitive\n * function that `visit` and `remote` calls when it receives a page.\n *\n * If you render a page outside the normal request response cycle, e.g,\n * websocket, you can use this function to save the payload.\n */\nexport function saveAndProcessPage(\n pageKey: string,\n page: PageResponse\n): SaveAndProcessPageThunk {\n return (dispatch) => {\n pageKey = urlToPageKey(pageKey)\n\n let nextPage = page\n\n page.fragments.reverse().forEach((fragment) => {\n const { id, path } = fragment\n const node = getIn(nextPage, path) as JSONMappable\n nextPage = setIn(page, path, { __id: id })\n\n dispatch(\n saveFragment({\n fragmentId: id,\n data: node,\n })\n )\n })\n\n if (nextPage.action === 'graft') {\n if (typeof nextPage.fragmentContext === 'string') {\n dispatch(\n handleFragmentGraft({\n fragmentId: nextPage.fragmentContext,\n response: nextPage,\n })\n )\n } else {\n dispatch(handleGraft({ pageKey, page: nextPage }))\n }\n } else if (nextPage.action === 'handleStreamResponse') {\n // We resolve the promise here because fragment responses\n // have deferment disabled.\n dispatch(handleStreamResponse(nextPage))\n return Promise.resolve()\n } else {\n dispatch(saveResponse({ pageKey, page: nextPage }))\n }\n\n const hasFetch = typeof fetch != 'undefined'\n if (hasFetch) {\n return dispatch(fetchDeferments(pageKey, nextPage.defers)).then(() =>\n Promise.resolve()\n )\n } else {\n return Promise.resolve()\n }\n }\n}\n"],"mappings":"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","names":["pathQuery","visit","remote"]}