|
| 1 | +/** |
| 2 | + * @jest-environment node |
| 3 | + */ |
| 4 | + |
1 | 5 | import type { StaticHandler, StaticHandlerContext } from "../router";
|
2 | 6 | import { UNSAFE_DEFERRED_SYMBOL, createStaticHandler } from "../router";
|
3 | 7 | import {
|
@@ -652,6 +656,106 @@ describe("ssr", () => {
|
652 | 656 | );
|
653 | 657 | });
|
654 | 658 |
|
| 659 | + it("should handle aborted load requests (v7_throwAbortReason=true)", async () => { |
| 660 | + let dfd = createDeferred(); |
| 661 | + let controller = new AbortController(); |
| 662 | + let { query } = createStaticHandler( |
| 663 | + [ |
| 664 | + { |
| 665 | + id: "root", |
| 666 | + path: "/path", |
| 667 | + loader: () => dfd.promise, |
| 668 | + }, |
| 669 | + ], |
| 670 | + { future: { v7_throwAbortReason: true } } |
| 671 | + ); |
| 672 | + let request = createRequest("/path?key=value", { |
| 673 | + signal: controller.signal, |
| 674 | + }); |
| 675 | + let e; |
| 676 | + try { |
| 677 | + let contextPromise = query(request); |
| 678 | + controller.abort(); |
| 679 | + // This should resolve even though we never resolved the loader |
| 680 | + await contextPromise; |
| 681 | + } catch (_e) { |
| 682 | + e = _e; |
| 683 | + } |
| 684 | + // DOMException added in node 17 |
| 685 | + if (process.versions.node.split(".").map(Number)[0] >= 17) { |
| 686 | + // eslint-disable-next-line jest/no-conditional-expect |
| 687 | + expect(e).toBeInstanceOf(DOMException); |
| 688 | + } |
| 689 | + expect(e.name).toBe("AbortError"); |
| 690 | + expect(e.message).toBe("This operation was aborted"); |
| 691 | + }); |
| 692 | + |
| 693 | + it("should handle aborted submit requests (v7_throwAbortReason=true)", async () => { |
| 694 | + let dfd = createDeferred(); |
| 695 | + let controller = new AbortController(); |
| 696 | + let { query } = createStaticHandler( |
| 697 | + [ |
| 698 | + { |
| 699 | + id: "root", |
| 700 | + path: "/path", |
| 701 | + action: () => dfd.promise, |
| 702 | + }, |
| 703 | + ], |
| 704 | + { future: { v7_throwAbortReason: true } } |
| 705 | + ); |
| 706 | + let request = createSubmitRequest("/path?key=value", { |
| 707 | + signal: controller.signal, |
| 708 | + }); |
| 709 | + let e; |
| 710 | + try { |
| 711 | + let contextPromise = query(request); |
| 712 | + controller.abort(); |
| 713 | + // This should resolve even though we never resolved the loader |
| 714 | + await contextPromise; |
| 715 | + } catch (_e) { |
| 716 | + e = _e; |
| 717 | + } |
| 718 | + // DOMException added in node 17 |
| 719 | + if (process.versions.node.split(".").map(Number)[0] >= 17) { |
| 720 | + // eslint-disable-next-line jest/no-conditional-expect |
| 721 | + expect(e).toBeInstanceOf(DOMException); |
| 722 | + } |
| 723 | + expect(e.name).toBe("AbortError"); |
| 724 | + expect(e.message).toBe("This operation was aborted"); |
| 725 | + }); |
| 726 | + |
| 727 | + it("should handle aborted requests (v7_throwAbortReason=true + custom reason)", async () => { |
| 728 | + let dfd = createDeferred(); |
| 729 | + let controller = new AbortController(); |
| 730 | + let { query } = createStaticHandler( |
| 731 | + [ |
| 732 | + { |
| 733 | + id: "root", |
| 734 | + path: "/path", |
| 735 | + loader: () => dfd.promise, |
| 736 | + }, |
| 737 | + ], |
| 738 | + { future: { v7_throwAbortReason: true } } |
| 739 | + ); |
| 740 | + let request = createRequest("/path?key=value", { |
| 741 | + signal: controller.signal, |
| 742 | + }); |
| 743 | + let e; |
| 744 | + try { |
| 745 | + let contextPromise = query(request); |
| 746 | + // Note this works in Node 18+ - but it does not work if using the |
| 747 | + // `abort-controller` polyfill which doesn't yet support a custom `reason` |
| 748 | + // See: https://github.com/mysticatea/abort-controller/issues/33 |
| 749 | + controller.abort(new Error("Oh no!")); |
| 750 | + // This should resolve even though we never resolved the loader |
| 751 | + await contextPromise; |
| 752 | + } catch (_e) { |
| 753 | + e = _e; |
| 754 | + } |
| 755 | + expect(e).toBeInstanceOf(Error); |
| 756 | + expect(e.message).toBe("Oh no!"); |
| 757 | + }); |
| 758 | + |
655 | 759 | it("should assign signals to requests by default (per the", async () => {
|
656 | 760 | let { query } = createStaticHandler(SSR_ROUTES);
|
657 | 761 | let request = createRequest("/", { signal: undefined });
|
@@ -1951,6 +2055,106 @@ describe("ssr", () => {
|
1951 | 2055 | );
|
1952 | 2056 | });
|
1953 | 2057 |
|
| 2058 | + it("should handle aborted load requests (v7_throwAbortReason=true)", async () => { |
| 2059 | + let dfd = createDeferred(); |
| 2060 | + let controller = new AbortController(); |
| 2061 | + let { queryRoute } = createStaticHandler( |
| 2062 | + [ |
| 2063 | + { |
| 2064 | + id: "root", |
| 2065 | + path: "/path", |
| 2066 | + loader: () => dfd.promise, |
| 2067 | + }, |
| 2068 | + ], |
| 2069 | + { future: { v7_throwAbortReason: true } } |
| 2070 | + ); |
| 2071 | + let request = createRequest("/path?key=value", { |
| 2072 | + signal: controller.signal, |
| 2073 | + }); |
| 2074 | + let e; |
| 2075 | + try { |
| 2076 | + let statePromise = queryRoute(request, { routeId: "root" }); |
| 2077 | + controller.abort(); |
| 2078 | + // This should resolve even though we never resolved the loader |
| 2079 | + await statePromise; |
| 2080 | + } catch (_e) { |
| 2081 | + e = _e; |
| 2082 | + } |
| 2083 | + // DOMException added in node 17 |
| 2084 | + if (process.versions.node.split(".").map(Number)[0] >= 17) { |
| 2085 | + // eslint-disable-next-line jest/no-conditional-expect |
| 2086 | + expect(e).toBeInstanceOf(DOMException); |
| 2087 | + } |
| 2088 | + expect(e.name).toBe("AbortError"); |
| 2089 | + expect(e.message).toBe("This operation was aborted"); |
| 2090 | + }); |
| 2091 | + |
| 2092 | + it("should handle aborted submit requests (v7_throwAbortReason=true)", async () => { |
| 2093 | + let dfd = createDeferred(); |
| 2094 | + let controller = new AbortController(); |
| 2095 | + let { queryRoute } = createStaticHandler( |
| 2096 | + [ |
| 2097 | + { |
| 2098 | + id: "root", |
| 2099 | + path: "/path", |
| 2100 | + action: () => dfd.promise, |
| 2101 | + }, |
| 2102 | + ], |
| 2103 | + { future: { v7_throwAbortReason: true } } |
| 2104 | + ); |
| 2105 | + let request = createSubmitRequest("/path?key=value", { |
| 2106 | + signal: controller.signal, |
| 2107 | + }); |
| 2108 | + let e; |
| 2109 | + try { |
| 2110 | + let statePromise = queryRoute(request, { routeId: "root" }); |
| 2111 | + controller.abort(); |
| 2112 | + // This should resolve even though we never resolved the loader |
| 2113 | + await statePromise; |
| 2114 | + } catch (_e) { |
| 2115 | + e = _e; |
| 2116 | + } |
| 2117 | + // DOMException added in node 17 |
| 2118 | + if (process.versions.node.split(".").map(Number)[0] >= 17) { |
| 2119 | + // eslint-disable-next-line jest/no-conditional-expect |
| 2120 | + expect(e).toBeInstanceOf(DOMException); |
| 2121 | + } |
| 2122 | + expect(e.name).toBe("AbortError"); |
| 2123 | + expect(e.message).toBe("This operation was aborted"); |
| 2124 | + }); |
| 2125 | + |
| 2126 | + it("should handle aborted load requests (v7_throwAbortReason=true + custom reason)", async () => { |
| 2127 | + let dfd = createDeferred(); |
| 2128 | + let controller = new AbortController(); |
| 2129 | + let { queryRoute } = createStaticHandler( |
| 2130 | + [ |
| 2131 | + { |
| 2132 | + id: "root", |
| 2133 | + path: "/path", |
| 2134 | + loader: () => dfd.promise, |
| 2135 | + }, |
| 2136 | + ], |
| 2137 | + { future: { v7_throwAbortReason: true } } |
| 2138 | + ); |
| 2139 | + let request = createRequest("/path?key=value", { |
| 2140 | + signal: controller.signal, |
| 2141 | + }); |
| 2142 | + let e; |
| 2143 | + try { |
| 2144 | + let statePromise = queryRoute(request, { routeId: "root" }); |
| 2145 | + // Note this works in Node 18+ - but it does not work if using the |
| 2146 | + // `abort-controller` polyfill which doesn't yet support a custom `reason` |
| 2147 | + // See: https://github.com/mysticatea/abort-controller/issues/33 |
| 2148 | + controller.abort(new Error("Oh no!")); |
| 2149 | + // This should resolve even though we never resolved the loader |
| 2150 | + await statePromise; |
| 2151 | + } catch (_e) { |
| 2152 | + e = _e; |
| 2153 | + } |
| 2154 | + expect(e).toBeInstanceOf(Error); |
| 2155 | + expect(e.message).toBe("Oh no!"); |
| 2156 | + }); |
| 2157 | + |
1954 | 2158 | it("should assign signals to requests by default (per the spec)", async () => {
|
1955 | 2159 | let { queryRoute } = createStaticHandler(SSR_ROUTES);
|
1956 | 2160 | let request = createRequest("/", { signal: undefined });
|
|
0 commit comments