@akinon/next 1.92.0-rc.32 → 1.92.0-rc.34

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/CHANGELOG.md CHANGED
@@ -1,5 +1,13 @@
1
1
  # @akinon/next
2
2
 
3
+ ## 1.92.0-rc.34
4
+
5
+ ### Minor Changes
6
+
7
+ - 0ad91bbd: ZERO-3489: Improve error handling in data fetching across multiple pages and server functions
8
+
9
+ ## 1.92.0-rc.33
10
+
3
11
  ## 1.92.0-rc.32
4
12
 
5
13
  ## 1.92.0-rc.31
@@ -5,7 +5,6 @@ import { category, product } from '../urls';
5
5
  import { Cache, CacheKey } from '../../lib/cache';
6
6
  import { parse } from 'lossless-json';
7
7
  import logger from '../../utils/log';
8
- import { headers as nHeaders } from 'next/headers';
9
8
  import { ServerVariables } from '../../utils/server-variables';
10
9
 
11
10
  function getCategoryDataHandler(
@@ -18,19 +17,30 @@ function getCategoryDataHandler(
18
17
  return async function () {
19
18
  const params = generateCommerceSearchParams(searchParams);
20
19
 
21
- const rawData = await appFetch<string>({
22
- url: `${category.getCategoryByPk(pk)}${params ? params : ''}`,
23
- locale,
24
- currency,
25
- init: {
26
- headers: {
27
- Accept: 'application/json',
28
- 'Content-Type': 'application/json',
29
- ...(headers ?? {})
30
- }
31
- },
32
- responseType: FetchResponseType.TEXT
33
- });
20
+ let rawData: string;
21
+
22
+ try {
23
+ rawData = await appFetch<string>({
24
+ url: `${category.getCategoryByPk(pk)}${params ? params : ''}`,
25
+ locale,
26
+ currency,
27
+ init: {
28
+ headers: {
29
+ Accept: 'application/json',
30
+ 'Content-Type': 'application/json',
31
+ ...(headers ?? {})
32
+ }
33
+ },
34
+ responseType: FetchResponseType.TEXT
35
+ });
36
+ } catch (error) {
37
+ logger.error('Failed to fetch category data', {
38
+ handler: 'getCategoryDataHandler',
39
+ pk,
40
+ error: error.message
41
+ });
42
+ return null;
43
+ }
34
44
 
35
45
  let data: GetCategoryResponse;
36
46
 
@@ -64,17 +74,27 @@ function getCategoryDataHandler(
64
74
  return { data, breadcrumbData: undefined };
65
75
  }
66
76
 
67
- const breadcrumbData = await appFetch<any>({
68
- url: product.breadcrumbUrl(menuItemModel),
69
- locale,
70
- currency,
71
- init: {
72
- headers: {
73
- Accept: 'application/json',
74
- 'Content-Type': 'application/json'
77
+ let breadcrumbData: { menu?: unknown } = {};
78
+
79
+ try {
80
+ breadcrumbData = await appFetch<{ menu?: unknown }>({
81
+ url: product.breadcrumbUrl(menuItemModel),
82
+ locale,
83
+ currency,
84
+ init: {
85
+ headers: {
86
+ Accept: 'application/json',
87
+ 'Content-Type': 'application/json'
88
+ }
75
89
  }
76
- }
77
- });
90
+ });
91
+ } catch (error) {
92
+ logger.warn('Failed to fetch breadcrumb data', {
93
+ handler: 'getCategoryDataHandler',
94
+ pk,
95
+ error: error.message
96
+ });
97
+ }
78
98
 
79
99
  return { data, breadcrumbData: breadcrumbData?.menu };
80
100
  };
@@ -11,20 +11,24 @@ const getFlatPageDataHandler = (
11
11
  headers?: Record<string, string>
12
12
  ) => {
13
13
  return async function () {
14
- const data = await appFetch<FlatPage>({
15
- url: flatpage.getFlatPageByPk(pk),
16
- locale,
17
- currency,
18
- init: {
19
- headers: {
20
- Accept: 'application/json',
21
- 'Content-Type': 'application/json',
22
- ...(headers ?? {})
14
+ try {
15
+ const data = await appFetch<FlatPage>({
16
+ url: flatpage.getFlatPageByPk(pk),
17
+ locale,
18
+ currency,
19
+ init: {
20
+ headers: {
21
+ Accept: 'application/json',
22
+ 'Content-Type': 'application/json',
23
+ ...(headers ?? {})
24
+ }
23
25
  }
24
- }
25
- });
26
+ });
26
27
 
27
- return data;
28
+ return data;
29
+ } catch (error) {
30
+ return null;
31
+ }
28
32
  };
29
33
  };
30
34
 
@@ -11,20 +11,24 @@ const getLandingPageHandler = (
11
11
  headers?: Record<string, string>
12
12
  ) => {
13
13
  return async function () {
14
- const data = await appFetch<LandingPage>({
15
- url: landingpage.getLandingPageByPk(pk),
16
- locale,
17
- currency,
18
- init: {
19
- headers: {
20
- Accept: 'application/json',
21
- 'Content-Type': 'application/json',
22
- ...(headers ?? {})
14
+ try {
15
+ const data = await appFetch<LandingPage>({
16
+ url: landingpage.getLandingPageByPk(pk),
17
+ locale,
18
+ currency,
19
+ init: {
20
+ headers: {
21
+ Accept: 'application/json',
22
+ 'Content-Type': 'application/json',
23
+ ...(headers ?? {})
24
+ }
23
25
  }
24
- }
25
- });
26
+ });
26
27
 
27
- return data;
28
+ return data;
29
+ } catch (error) {
30
+ return null;
31
+ }
28
32
  };
29
33
  };
30
34
 
@@ -16,19 +16,29 @@ const getListDataHandler = (
16
16
  return async function () {
17
17
  const params = generateCommerceSearchParams(searchParams);
18
18
 
19
- const rawData = await appFetch<string>({
20
- url: `${category.list}${params}`,
21
- locale,
22
- currency,
23
- init: {
24
- headers: {
25
- Accept: 'application/json',
26
- 'Content-Type': 'application/json',
27
- ...(headers ?? {})
28
- }
29
- },
30
- responseType: FetchResponseType.TEXT
31
- });
19
+ let rawData: string;
20
+
21
+ try {
22
+ rawData = await appFetch<string>({
23
+ url: `${category.list}${params}`,
24
+ locale,
25
+ currency,
26
+ init: {
27
+ headers: {
28
+ Accept: 'application/json',
29
+ 'Content-Type': 'application/json',
30
+ ...(headers ?? {})
31
+ }
32
+ },
33
+ responseType: FetchResponseType.TEXT
34
+ });
35
+ } catch (error) {
36
+ logger.error('Failed to fetch list data', {
37
+ handler: 'getListDataHandler',
38
+ error: error.message
39
+ });
40
+ return null;
41
+ }
32
42
 
33
43
  let data: GetCategoryResponse;
34
44
 
@@ -35,57 +35,84 @@ const getProductDataHandler = ({
35
35
  .join('&');
36
36
  }
37
37
 
38
- const data = await appFetch<ProductResult>({
39
- url,
40
- locale,
41
- currency,
42
- init: {
43
- headers: {
44
- Accept: 'application/json',
45
- 'Content-Type': 'application/json',
46
- ...(headers ?? {})
38
+ let data: ProductResult;
39
+
40
+ try {
41
+ data = await appFetch<ProductResult>({
42
+ url,
43
+ locale,
44
+ currency,
45
+ init: {
46
+ headers: {
47
+ Accept: 'application/json',
48
+ 'Content-Type': 'application/json',
49
+ ...(headers ?? {})
50
+ }
47
51
  }
48
- }
49
- });
52
+ });
53
+ } catch (error) {
54
+ logger.error('Failed to fetch product data', {
55
+ handler: 'getProductDataHandler',
56
+ pk,
57
+ error: error.message,
58
+ url
59
+ });
60
+ return null;
61
+ }
50
62
 
51
63
  const categoryUrl = product.categoryUrl(data.product.pk);
52
64
 
53
- const productCategoryData = await appFetch<ProductCategoryResult>({
54
- url: categoryUrl,
55
- locale,
56
- currency,
57
- init: {
58
- headers: {
59
- Accept: 'application/json',
60
- 'Content-Type': 'application/json'
65
+ let productCategoryData: ProductCategoryResult;
66
+ let breadcrumbData: { menu?: unknown } = {};
67
+
68
+ try {
69
+ productCategoryData = await appFetch<ProductCategoryResult>({
70
+ url: categoryUrl,
71
+ locale,
72
+ currency,
73
+ init: {
74
+ headers: {
75
+ Accept: 'application/json',
76
+ 'Content-Type': 'application/json'
77
+ }
61
78
  }
79
+ });
80
+
81
+ const menuItemModel = productCategoryData?.results[0]?.menuitemmodel;
82
+
83
+ if (!menuItemModel) {
84
+ logger.warn(
85
+ 'menuItemModel is undefined, skipping breadcrumbData fetch',
86
+ {
87
+ handler: 'getProductDataHandler',
88
+ pk
89
+ }
90
+ );
91
+ return { data, breadcrumbData: undefined };
62
92
  }
63
- });
64
93
 
65
- const menuItemModel = productCategoryData?.results[0]?.menuitemmodel;
94
+ const breadcrumbUrl = product.breadcrumbUrl(menuItemModel);
66
95
 
67
- if (!menuItemModel) {
68
- logger.warn('menuItemModel is undefined, skipping breadcrumbData fetch', {
96
+ breadcrumbData = await appFetch<{ menu?: unknown }>({
97
+ url: breadcrumbUrl,
98
+ locale,
99
+ currency,
100
+ init: {
101
+ headers: {
102
+ Accept: 'application/json',
103
+ 'Content-Type': 'application/json'
104
+ }
105
+ }
106
+ });
107
+ } catch (error) {
108
+ logger.warn('Failed to fetch breadcrumb data', {
69
109
  handler: 'getProductDataHandler',
70
- pk
110
+ pk,
111
+ error: error.message
71
112
  });
72
- return { data, breadcrumbData: undefined };
113
+ // Continue without breadcrumb data
73
114
  }
74
115
 
75
- const breadcrumbUrl = product.breadcrumbUrl(menuItemModel);
76
-
77
- const breadcrumbData = await appFetch<any>({
78
- url: breadcrumbUrl,
79
- locale,
80
- currency,
81
- init: {
82
- headers: {
83
- Accept: 'application/json',
84
- 'Content-Type': 'application/json'
85
- }
86
- }
87
- });
88
-
89
116
  return {
90
117
  data,
91
118
  breadcrumbData: breadcrumbData?.menu
@@ -15,20 +15,24 @@ const getSpecialPageDataHandler = (
15
15
  return async function () {
16
16
  const params = generateCommerceSearchParams(searchParams);
17
17
 
18
- const data: GetCategoryResponse = await appFetch({
19
- url: `${category.getSpecialPageByPk(pk)}${params}`,
20
- locale,
21
- currency,
22
- init: {
23
- headers: {
24
- Accept: 'application/json',
25
- 'Content-Type': 'application/json',
26
- ...(headers ?? {})
18
+ try {
19
+ const data: GetCategoryResponse = await appFetch({
20
+ url: `${category.getSpecialPageByPk(pk)}${params}`,
21
+ locale,
22
+ currency,
23
+ init: {
24
+ headers: {
25
+ Accept: 'application/json',
26
+ 'Content-Type': 'application/json',
27
+ ...(headers ?? {})
28
+ }
27
29
  }
28
- }
29
- });
30
+ });
30
31
 
31
- return data;
32
+ return data;
33
+ } catch (error) {
34
+ return null;
35
+ }
32
36
  };
33
37
  };
34
38
 
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@akinon/next",
3
3
  "description": "Core package for Project Zero Next",
4
- "version": "1.92.0-rc.32",
4
+ "version": "1.92.0-rc.34",
5
5
  "private": false,
6
6
  "license": "MIT",
7
7
  "bin": {
@@ -34,7 +34,7 @@
34
34
  "set-cookie-parser": "2.6.0"
35
35
  },
36
36
  "devDependencies": {
37
- "@akinon/eslint-plugin-projectzero": "1.92.0-rc.32",
37
+ "@akinon/eslint-plugin-projectzero": "1.92.0-rc.34",
38
38
  "@babel/core": "7.26.10",
39
39
  "@babel/preset-env": "7.26.9",
40
40
  "@babel/preset-typescript": "7.27.0",
@@ -60,6 +60,11 @@ const appFetch = async <T>({
60
60
  status = req.status;
61
61
  logger.debug(`FETCH END ${url}`, { status: req.status, ip });
62
62
 
63
+ if (!req.ok) {
64
+ const errorMessage = `HTTP ${req.status}: ${req.statusText}`;
65
+ throw new Error(errorMessage);
66
+ }
67
+
63
68
  if (responseType === FetchResponseType.JSON) {
64
69
  response = (await req.json()) as T;
65
70
  } else {