@@ -69,3 +69,123 @@ test('json data rewrite works', async ({ middlewarePages }) => {
69
69
70
70
expect ( data . pageProps . message ) . toBeDefined ( )
71
71
} )
72
+
73
+ // those tests use `fetch` instead of `page.goto` intentionally to avoid potential client rendering
74
+ // hiding any potential edge/server issues
75
+ test . describe ( 'Middleware with i18n and excluding paths' , ( ) => {
76
+ const DEFAULT_LOCALE = 'en'
77
+
78
+ // those tests hit paths ending with `/json` which has special handling in middleware
79
+ // to return JSON response from middleware itself
80
+ test . describe ( 'Middleware response path' , ( ) => {
81
+ test ( 'should match on non-localized not excluded path' , async ( {
82
+ middlewareI18nExcludedPaths,
83
+ } ) => {
84
+ const response = await fetch ( `${ middlewareI18nExcludedPaths . url } /json` )
85
+
86
+ expect ( response . headers . get ( 'x-test-used-middleware' ) ) . toBe ( 'true' )
87
+ expect ( response . status ) . toBe ( 200 )
88
+
89
+ const { nextUrlPathname, nextUrlLocale } = await response . json ( )
90
+
91
+ expect ( nextUrlPathname ) . toBe ( '/json' )
92
+ expect ( nextUrlLocale ) . toBe ( DEFAULT_LOCALE )
93
+ } )
94
+
95
+ test ( 'should match on localized not excluded path' , async ( { middlewareI18nExcludedPaths } ) => {
96
+ const response = await fetch ( `${ middlewareI18nExcludedPaths . url } /fr/json` )
97
+
98
+ expect ( response . headers . get ( 'x-test-used-middleware' ) ) . toBe ( 'true' )
99
+ expect ( response . status ) . toBe ( 200 )
100
+
101
+ const { nextUrlPathname, nextUrlLocale } = await response . json ( )
102
+
103
+ expect ( nextUrlPathname ) . toBe ( '/json' )
104
+ expect ( nextUrlLocale ) . toBe ( 'fr' )
105
+ } )
106
+
107
+ test ( 'should NOT match on non-localized excluded path' , async ( {
108
+ middlewareI18nExcludedPaths,
109
+ } ) => {
110
+ const response = await fetch ( `${ middlewareI18nExcludedPaths . url } /api/json` )
111
+
112
+ expect ( response . headers . get ( 'x-test-used-middleware' ) ) . not . toBe ( 'true' )
113
+
114
+ // we are not matching middleware, so we will be served raw JSON response from `pages/api/[[...catchall]].ts`
115
+ expect ( response . status ) . toBe ( 200 )
116
+
117
+ const { params } = await response . json ( )
118
+
119
+ expect ( params ) . toMatchObject ( { catchall : [ 'json' ] } )
120
+ } )
121
+ } )
122
+
123
+ // those tests hit paths that don't end with `/json` so they should be passed through to origin
124
+ test . describe ( 'Middleware passthrough' , ( ) => {
125
+ function extractDataFromHtml ( html : string ) : Record < string , any > {
126
+ const match = html . match ( / < p r e > (?< rawInput > [ ^ < ] + ) < \/ p r e > / )
127
+ if ( ! match || ! match . groups ?. rawInput ) {
128
+ console . error ( '<pre> not found in html input' , {
129
+ html,
130
+ } )
131
+ throw new Error ( 'Failed to extract data from HTML' )
132
+ }
133
+
134
+ const { rawInput } = match . groups
135
+ const unescapedInput = rawInput . replaceAll ( '"' , '"' )
136
+ try {
137
+ return JSON . parse ( unescapedInput )
138
+ } catch ( originalError ) {
139
+ console . error ( 'Failed to parse JSON' , {
140
+ originalError,
141
+ rawInput,
142
+ unescapedInput,
143
+ } )
144
+ }
145
+ throw new Error ( 'Failed to extract data from HTML' )
146
+ }
147
+
148
+ test ( 'should match on non-localized not excluded path' , async ( {
149
+ middlewareI18nExcludedPaths,
150
+ } ) => {
151
+ const response = await fetch ( `${ middlewareI18nExcludedPaths . url } /html` )
152
+
153
+ expect ( response . headers . get ( 'x-test-used-middleware' ) ) . toBe ( 'true' )
154
+ expect ( response . status ) . toBe ( 200 )
155
+ expect ( response . headers . get ( 'content-type' ) ) . toMatch ( / t e x t \/ h t m l / )
156
+
157
+ const html = await response . text ( )
158
+ const { locale, params } = extractDataFromHtml ( html )
159
+
160
+ expect ( params ) . toMatchObject ( { catchall : [ 'html' ] } )
161
+ expect ( locale ) . toBe ( DEFAULT_LOCALE )
162
+ } )
163
+
164
+ test ( 'should match on localized not excluded path' , async ( { middlewareI18nExcludedPaths } ) => {
165
+ const response = await fetch ( `${ middlewareI18nExcludedPaths . url } /fr/html` )
166
+
167
+ expect ( response . headers . get ( 'x-test-used-middleware' ) ) . toBe ( 'true' )
168
+ expect ( response . status ) . toBe ( 200 )
169
+ expect ( response . headers . get ( 'content-type' ) ) . toMatch ( / t e x t \/ h t m l / )
170
+
171
+ const html = await response . text ( )
172
+ const { locale, params } = extractDataFromHtml ( html )
173
+
174
+ expect ( params ) . toMatchObject ( { catchall : [ 'html' ] } )
175
+ expect ( locale ) . toBe ( 'fr' )
176
+ } )
177
+
178
+ test ( 'should NOT match on non-localized excluded path' , async ( {
179
+ middlewareI18nExcludedPaths,
180
+ } ) => {
181
+ const response = await fetch ( `${ middlewareI18nExcludedPaths . url } /api/html` )
182
+
183
+ expect ( response . headers . get ( 'x-test-used-middleware' ) ) . not . toBe ( 'true' )
184
+ expect ( response . status ) . toBe ( 200 )
185
+
186
+ const { params } = await response . json ( )
187
+
188
+ expect ( params ) . toMatchObject ( { catchall : [ 'html' ] } )
189
+ } )
190
+ } )
191
+ } )
0 commit comments