czon 0.1.0 → 0.1.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.
Files changed (263) hide show
  1. package/.github/workflows/bump-version.yml +112 -0
  2. package/.github/workflows/ci.yml +64 -0
  3. package/.github/workflows/pages.yml +61 -0
  4. package/.github/workflows/publish.yml +81 -0
  5. package/.husky/pre-commit +1 -0
  6. package/.prettierignore +14 -0
  7. package/.prettierrc.json +11 -0
  8. package/.zen/meta.json +106 -0
  9. package/.zen/src/ar-SA/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  10. package/.zen/src/ar-SA/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  11. package/.zen/src/ar-SA/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  12. package/.zen/src/ar-SA/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  13. package/.zen/src/ar-SA/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  14. package/.zen/src/ar-SA/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  15. package/.zen/src/da-DK/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  16. package/.zen/src/da-DK/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  17. package/.zen/src/da-DK/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  18. package/.zen/src/da-DK/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  19. package/.zen/src/da-DK/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  20. package/.zen/src/da-DK/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  21. package/.zen/src/de-DE/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  22. package/.zen/src/de-DE/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  23. package/.zen/src/de-DE/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  24. package/.zen/src/de-DE/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  25. package/.zen/src/de-DE/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  26. package/.zen/src/de-DE/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  27. package/.zen/src/en-US/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  28. package/.zen/src/en-US/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  29. package/.zen/src/en-US/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  30. package/.zen/src/en-US/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  31. package/.zen/src/en-US/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  32. package/.zen/src/en-US/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  33. package/.zen/src/es-ES/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  34. package/.zen/src/es-ES/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  35. package/.zen/src/es-ES/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  36. package/.zen/src/es-ES/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  37. package/.zen/src/es-ES/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  38. package/.zen/src/es-ES/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  39. package/.zen/src/es-MX/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  40. package/.zen/src/es-MX/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  41. package/.zen/src/es-MX/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  42. package/.zen/src/es-MX/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  43. package/.zen/src/es-MX/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  44. package/.zen/src/es-MX/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  45. package/.zen/src/fi-FI/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  46. package/.zen/src/fi-FI/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  47. package/.zen/src/fi-FI/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  48. package/.zen/src/fi-FI/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  49. package/.zen/src/fi-FI/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  50. package/.zen/src/fi-FI/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  51. package/.zen/src/fr-FR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  52. package/.zen/src/fr-FR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  53. package/.zen/src/fr-FR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  54. package/.zen/src/fr-FR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  55. package/.zen/src/fr-FR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  56. package/.zen/src/fr-FR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  57. package/.zen/src/hi-IN/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  58. package/.zen/src/hi-IN/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  59. package/.zen/src/hi-IN/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  60. package/.zen/src/hi-IN/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  61. package/.zen/src/hi-IN/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  62. package/.zen/src/hi-IN/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  63. package/.zen/src/id-ID/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  64. package/.zen/src/id-ID/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  65. package/.zen/src/id-ID/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  66. package/.zen/src/id-ID/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  67. package/.zen/src/id-ID/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  68. package/.zen/src/id-ID/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  69. package/.zen/src/it-IT/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  70. package/.zen/src/it-IT/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  71. package/.zen/src/it-IT/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  72. package/.zen/src/it-IT/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  73. package/.zen/src/it-IT/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  74. package/.zen/src/it-IT/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  75. package/.zen/src/ja-JP/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  76. package/.zen/src/ja-JP/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  77. package/.zen/src/ja-JP/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  78. package/.zen/src/ja-JP/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  79. package/.zen/src/ja-JP/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  80. package/.zen/src/ja-JP/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  81. package/.zen/src/ko-KR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  82. package/.zen/src/ko-KR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  83. package/.zen/src/ko-KR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  84. package/.zen/src/ko-KR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  85. package/.zen/src/ko-KR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  86. package/.zen/src/ko-KR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  87. package/.zen/src/nl-NL/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  88. package/.zen/src/nl-NL/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  89. package/.zen/src/nl-NL/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  90. package/.zen/src/nl-NL/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  91. package/.zen/src/nl-NL/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  92. package/.zen/src/nl-NL/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  93. package/.zen/src/no-NO/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  94. package/.zen/src/no-NO/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  95. package/.zen/src/no-NO/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  96. package/.zen/src/no-NO/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  97. package/.zen/src/no-NO/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  98. package/.zen/src/no-NO/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  99. package/.zen/src/pl-PL/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  100. package/.zen/src/pl-PL/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  101. package/.zen/src/pl-PL/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  102. package/.zen/src/pl-PL/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  103. package/.zen/src/pl-PL/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  104. package/.zen/src/pl-PL/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  105. package/.zen/src/pt-BR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  106. package/.zen/src/pt-BR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  107. package/.zen/src/pt-BR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  108. package/.zen/src/pt-BR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  109. package/.zen/src/pt-BR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  110. package/.zen/src/pt-BR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  111. package/.zen/src/pt-PT/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  112. package/.zen/src/pt-PT/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  113. package/.zen/src/pt-PT/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  114. package/.zen/src/pt-PT/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  115. package/.zen/src/pt-PT/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  116. package/.zen/src/pt-PT/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  117. package/.zen/src/ru-RU/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  118. package/.zen/src/ru-RU/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  119. package/.zen/src/ru-RU/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  120. package/.zen/src/ru-RU/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  121. package/.zen/src/ru-RU/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  122. package/.zen/src/ru-RU/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  123. package/.zen/src/sv-SE/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  124. package/.zen/src/sv-SE/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  125. package/.zen/src/sv-SE/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  126. package/.zen/src/sv-SE/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  127. package/.zen/src/sv-SE/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  128. package/.zen/src/sv-SE/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  129. package/.zen/src/th-TH/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  130. package/.zen/src/th-TH/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  131. package/.zen/src/th-TH/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  132. package/.zen/src/th-TH/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  133. package/.zen/src/th-TH/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  134. package/.zen/src/th-TH/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  135. package/.zen/src/tr-TR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  136. package/.zen/src/tr-TR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  137. package/.zen/src/tr-TR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  138. package/.zen/src/tr-TR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  139. package/.zen/src/tr-TR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  140. package/.zen/src/tr-TR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  141. package/.zen/src/uk-UA/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  142. package/.zen/src/uk-UA/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +79 -0
  143. package/.zen/src/uk-UA/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  144. package/.zen/src/uk-UA/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  145. package/.zen/src/uk-UA/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  146. package/.zen/src/uk-UA/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  147. package/.zen/src/vi-VN/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +121 -0
  148. package/.zen/src/vi-VN/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +78 -0
  149. package/.zen/src/vi-VN/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  150. package/.zen/src/vi-VN/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +71 -0
  151. package/.zen/src/vi-VN/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  152. package/.zen/src/vi-VN/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +30 -0
  153. package/.zen/src/zh-Hans/4b2cedd76dddb2a20b1890460735dd38fa65e293a37d6bd15310852ba1c851c4.md +21 -0
  154. package/.zen/src/zh-Hans/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +123 -0
  155. package/.zen/src/zh-Hans/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +81 -0
  156. package/.zen/src/zh-Hans/59fe7aff1dcf6f4bbbaff1ad2f7c4c5a3d34b99b1d288b78717ee34525c9897b.md +24 -0
  157. package/.zen/src/zh-Hans/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +74 -0
  158. package/.zen/src/zh-Hans/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +74 -0
  159. package/.zen/src/zh-Hans/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +75 -0
  160. package/.zen/src/zh-Hans/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +31 -0
  161. package/.zen/src/zh-Hans/c1b0e9eb01d78140c21097187d6395815fd01a6f179af31b7e72532ec6c7c682.md +75 -0
  162. package/.zen/src/zh-Hans/cbf2fcda767ee28fe49828908c14de2c8edba8f8b7d721743eda767838b70923.md +67 -0
  163. package/.zen/src/zh-Hans/ec9fa686ad66e28d8d9ee131812bb6f31052185584de0853db1ee8893ebddda3.md +22 -0
  164. package/.zen/src/zh-Hans/f119cbaf7645c713b1e0fa67abdc1070766c80f94eec9e42108bfa37ef85db43.md +24 -0
  165. package/.zen/src/zh-Hant/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +123 -0
  166. package/.zen/src/zh-Hant/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +81 -0
  167. package/.zen/src/zh-Hant/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +70 -0
  168. package/.zen/src/zh-Hant/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +74 -0
  169. package/.zen/src/zh-Hant/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +71 -0
  170. package/.zen/src/zh-Hant/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +31 -0
  171. package/README.md +56 -0
  172. package/assets/templates/default/layout.html +421 -0
  173. package/dist/ai/extractMetadataFromMarkdown.d.ts +8 -0
  174. package/dist/ai/extractMetadataFromMarkdown.d.ts.map +1 -0
  175. package/dist/ai/extractMetadataFromMarkdown.js +88 -0
  176. package/dist/ai/extractMetadataFromMarkdown.js.map +1 -0
  177. package/dist/ai/translateMarkdown.d.ts +8 -0
  178. package/dist/ai/translateMarkdown.d.ts.map +1 -0
  179. package/dist/ai/translateMarkdown.js +60 -0
  180. package/dist/ai/translateMarkdown.js.map +1 -0
  181. package/dist/build/pipeline.d.ts +6 -0
  182. package/dist/build/pipeline.d.ts.map +1 -0
  183. package/dist/build/pipeline.js +180 -0
  184. package/dist/build/pipeline.js.map +1 -0
  185. package/dist/cli.d.ts +3 -0
  186. package/dist/cli.d.ts.map +1 -0
  187. package/dist/cli.js +102 -0
  188. package/dist/cli.js.map +1 -0
  189. package/dist/findEntries.d.ts +10 -0
  190. package/dist/findEntries.d.ts.map +1 -0
  191. package/dist/findEntries.js +38 -0
  192. package/dist/findEntries.js.map +1 -0
  193. package/dist/index.d.ts +2 -0
  194. package/dist/index.d.ts.map +1 -0
  195. package/dist/index.js +4 -0
  196. package/dist/index.js.map +1 -0
  197. package/dist/languages.d.ts +2 -0
  198. package/dist/languages.d.ts.map +1 -0
  199. package/dist/languages.js +37 -0
  200. package/dist/languages.js.map +1 -0
  201. package/dist/metadata.d.ts +14 -0
  202. package/dist/metadata.d.ts.map +1 -0
  203. package/dist/metadata.js +78 -0
  204. package/dist/metadata.js.map +1 -0
  205. package/dist/paths.d.ts +6 -0
  206. package/dist/paths.d.ts.map +1 -0
  207. package/dist/paths.js +10 -0
  208. package/dist/paths.js.map +1 -0
  209. package/dist/process/extractMetadataByAI.d.ts +5 -0
  210. package/dist/process/extractMetadataByAI.d.ts.map +1 -0
  211. package/dist/process/extractMetadataByAI.js +31 -0
  212. package/dist/process/extractMetadataByAI.js.map +1 -0
  213. package/dist/process/scanSourceFiles.d.ts +5 -0
  214. package/dist/process/scanSourceFiles.d.ts.map +1 -0
  215. package/dist/process/scanSourceFiles.js +70 -0
  216. package/dist/process/scanSourceFiles.js.map +1 -0
  217. package/dist/process/template.d.ts +5 -0
  218. package/dist/process/template.d.ts.map +1 -0
  219. package/dist/process/template.js +207 -0
  220. package/dist/process/template.js.map +1 -0
  221. package/dist/services/openai.d.ts +41 -0
  222. package/dist/services/openai.d.ts.map +1 -0
  223. package/dist/services/openai.js +54 -0
  224. package/dist/services/openai.js.map +1 -0
  225. package/dist/types.d.ts +35 -0
  226. package/dist/types.d.ts.map +1 -0
  227. package/dist/types.js +3 -0
  228. package/dist/types.js.map +1 -0
  229. package/dist/utils/convertMarkdownToHtml.d.ts +7 -0
  230. package/dist/utils/convertMarkdownToHtml.d.ts.map +1 -0
  231. package/dist/utils/convertMarkdownToHtml.js +94 -0
  232. package/dist/utils/convertMarkdownToHtml.js.map +1 -0
  233. package/dist/utils/frontmatter.d.ts +6 -0
  234. package/dist/utils/frontmatter.d.ts.map +1 -0
  235. package/dist/utils/frontmatter.js +22 -0
  236. package/dist/utils/frontmatter.js.map +1 -0
  237. package/dist/utils/sha256.d.ts +2 -0
  238. package/dist/utils/sha256.d.ts.map +1 -0
  239. package/dist/utils/sha256.js +7 -0
  240. package/dist/utils/sha256.js.map +1 -0
  241. package/docs/test-katex.md +66 -0
  242. package/docs/test-mermaid.md +105 -0
  243. package/docs/tricks.md +15 -0
  244. package/package.json +58 -6
  245. package/src/ai/extractMetadataFromMarkdown.ts +95 -0
  246. package/src/ai/translateMarkdown.ts +60 -0
  247. package/src/build/pipeline.ts +172 -0
  248. package/src/cli.ts +73 -0
  249. package/src/findEntries.ts +37 -0
  250. package/src/index.ts +1 -0
  251. package/src/languages.ts +37 -0
  252. package/src/metadata.ts +44 -0
  253. package/src/paths.ts +7 -0
  254. package/src/process/extractMetadataByAI.ts +31 -0
  255. package/src/process/scanSourceFiles.ts +73 -0
  256. package/src/process/template.ts +222 -0
  257. package/src/services/openai.ts +92 -0
  258. package/src/types.ts +37 -0
  259. package/src/utils/convertMarkdownToHtml.ts +93 -0
  260. package/src/utils/frontmatter.ts +18 -0
  261. package/src/utils/sha256.ts +4 -0
  262. package/test-multilang.js +44 -0
  263. package/tsconfig.json +19 -0
@@ -0,0 +1,37 @@
1
+ export const LANGUAGE_NAMES: Record<string, string> = {
2
+ // 现有语言
3
+ 'zh-Hans': '简体中文',
4
+ 'en-US': 'English',
5
+ 'ja-JP': '日本語',
6
+ 'ko-KR': '한국어',
7
+ 'es-ES': 'Español',
8
+ 'fr-FR': 'Français',
9
+ 'de-DE': 'Deutsch',
10
+ 'ru-RU': 'Русский',
11
+
12
+ // 补充缺失的项目支持语言
13
+ 'pt-PT': 'Português',
14
+
15
+ // 欧洲主要语言
16
+ 'it-IT': 'Italiano',
17
+ 'nl-NL': 'Nederlands',
18
+ 'pl-PL': 'Polski',
19
+ 'sv-SE': 'Svenska',
20
+ 'fi-FI': 'Suomi',
21
+ 'da-DK': 'Dansk',
22
+ 'no-NO': 'Norsk',
23
+
24
+ // 亚洲主要语言
25
+ 'zh-Hant': '繁體中文',
26
+ 'hi-IN': 'हिन्दी',
27
+ 'ar-SA': 'العربية',
28
+ 'th-TH': 'ไทย',
29
+ 'vi-VN': 'Tiếng Việt',
30
+ 'id-ID': 'Bahasa Indonesia',
31
+
32
+ // 其他重要语言
33
+ 'pt-BR': 'Português (Brasil)',
34
+ 'es-MX': 'Español (México)',
35
+ 'tr-TR': 'Türkçe',
36
+ 'uk-UA': 'Українська',
37
+ };
@@ -0,0 +1,44 @@
1
+ import * as fs from 'fs/promises';
2
+ import * as path from 'path';
3
+ import { ZEN_META_PATH } from './paths';
4
+ import { MetaDataStore } from './types';
5
+
6
+ /**
7
+ * 全局 MetaDataStore 单例
8
+ */
9
+ export const MetaData: MetaDataStore = {
10
+ // 稍后覆盖
11
+ version: '1.0.0',
12
+ options: {},
13
+ files: [],
14
+ };
15
+
16
+ /**
17
+ * 从文件中读取数据,覆盖 store,但是要保持它仍然是同一个对象
18
+ */
19
+ export async function loadMetaData(): Promise<void> {
20
+ try {
21
+ await fs.access(ZEN_META_PATH);
22
+ const content = await fs.readFile(ZEN_META_PATH, 'utf-8');
23
+ const newData = JSON.parse(content);
24
+
25
+ // 使用 Object.assign 保持同一个对象引用
26
+ Object.assign(MetaData, newData);
27
+ } catch (error) {
28
+ // 如果文件不存在,初始化默认值
29
+ MetaData.version = '1.0.0';
30
+ MetaData.files = [];
31
+ }
32
+ }
33
+
34
+ /**
35
+ * 将 MetaData 写入 store
36
+ */
37
+ export async function saveMetaData(): Promise<void> {
38
+ // 确保 .zen 目录存在
39
+ const zenDir = path.dirname(ZEN_META_PATH);
40
+ await fs.mkdir(zenDir, { recursive: true });
41
+
42
+ // 保存文件
43
+ await fs.writeFile(ZEN_META_PATH, JSON.stringify(MetaData, null, 2), 'utf-8');
44
+ }
package/src/paths.ts ADDED
@@ -0,0 +1,7 @@
1
+ import { join } from 'path';
2
+
3
+ export const INPUT_DIR = process.cwd();
4
+ export const ZEN_DIR = join(process.cwd(), '.zen');
5
+ export const ZEN_DIST_DIR = join(ZEN_DIR, 'dist');
6
+ export const ZEN_SRC_DIR = join(ZEN_DIR, 'src');
7
+ export const ZEN_META_PATH = join(ZEN_DIR, 'meta.json');
@@ -0,0 +1,31 @@
1
+ import { readFile } from 'fs/promises';
2
+ import { extractMetadataFromMarkdown } from '../ai/extractMetadataFromMarkdown';
3
+ import { MetaData } from '../metadata';
4
+
5
+ /**
6
+ * 运行 AI 元数据提取
7
+ */
8
+ export async function extractMetadataByAI(): Promise<void> {
9
+ const { files } = MetaData;
10
+
11
+ if (MetaData.options.verbose) console.log(`🤖 Running AI metadata extraction...`);
12
+ console.log(`🤖 Processing ${files.length} files with AI...`);
13
+
14
+ await Promise.all(
15
+ files.map(async file => {
16
+ try {
17
+ if (file.metadata) {
18
+ console.info(`ℹ️ Skipping ${file.path}, already has metadata`);
19
+ return;
20
+ }
21
+ const content = await readFile(file.path, 'utf-8');
22
+ file.metadata = await extractMetadataFromMarkdown(content);
23
+ console.log(`✅ Extracted AI metadata for ${file.path}`, file.metadata.tokens_used);
24
+ } catch (error) {
25
+ console.error(`⚠️ Failed to process file ${file.path}:`, error);
26
+ }
27
+ })
28
+ );
29
+
30
+ console.log(`✅ AI processing completed for ${files.length} files`);
31
+ }
@@ -0,0 +1,73 @@
1
+ import { readFile } from 'fs/promises';
2
+ import path from 'path';
3
+ import { findMarkdownEntries } from '../findEntries';
4
+ import { MetaData } from '../metadata';
5
+ import { INPUT_DIR } from '../paths';
6
+ import { sha256 } from '../utils/sha256';
7
+
8
+ const extractLinksFromMarkdown = (content: string): string[] => {
9
+ const linkRegex = /\[.*?\]\((.*?)\)/g;
10
+ const links: string[] = [];
11
+ let match;
12
+ while ((match = linkRegex.exec(content)) !== null) {
13
+ links.push(match[1]);
14
+ }
15
+ return links;
16
+ };
17
+
18
+ /**
19
+ * 扫描源文件
20
+ */
21
+ export async function scanSourceFiles(): Promise<void> {
22
+ console.log(`🔍 Scanning source directory...`);
23
+ const markdownFiles = await findMarkdownEntries(INPUT_DIR);
24
+ const hashes = new Set<string>();
25
+
26
+ for (const relativePath of markdownFiles) {
27
+ const fullPath = path.join(INPUT_DIR, relativePath);
28
+
29
+ try {
30
+ // 检查文件是否存在
31
+
32
+ const content = await readFile(fullPath, 'utf-8'); // 确保文件可读
33
+
34
+ const hash = sha256(content);
35
+ const links = extractLinksFromMarkdown(content);
36
+ console.info(` - Found file: ${relativePath} (hash: ${hash})`);
37
+ console.info(` Links: ${links.join(', ') || 'None'}`);
38
+
39
+ hashes.add(hash);
40
+
41
+ const metaWithSameHash = MetaData.files.find(f => f.hash === hash);
42
+ if (metaWithSameHash) {
43
+ metaWithSameHash.path = relativePath;
44
+ metaWithSameHash.links = links;
45
+ } else {
46
+ // 如果没有相同哈希的元数据,则添加一个新的占位符
47
+ MetaData.files.push({
48
+ hash,
49
+ path: relativePath,
50
+ links,
51
+ });
52
+ }
53
+ } catch (error) {
54
+ console.warn(`⚠️ File not found or inaccessible: ${fullPath}`, error);
55
+ }
56
+ }
57
+ // 移除不再存在的文件元数据
58
+ MetaData.files = MetaData.files.filter(f => hashes.has(f.hash));
59
+ // 按路径降序排序 (通常外层目录优先)
60
+ MetaData.files.sort(
61
+ (a, b) =>
62
+ // 第一级按目录排序
63
+ path.dirname(a.path).localeCompare(path.dirname(b.path)) ||
64
+ // 第二级按文件名排序
65
+ a.path.localeCompare(b.path)
66
+ );
67
+
68
+ console.log(`✅ Found ${MetaData.files.length} Markdown files`);
69
+
70
+ if (MetaData.files.length === 0) {
71
+ console.warn(`⚠️ No Markdown files found in ${INPUT_DIR}`);
72
+ }
73
+ }
@@ -0,0 +1,222 @@
1
+ import * as fs from 'fs/promises';
2
+ import * as path from 'path';
3
+ import { LANGUAGE_NAMES } from '../languages';
4
+ import { MetaData } from '../metadata';
5
+ import { ZEN_DIST_DIR, ZEN_SRC_DIR } from '../paths';
6
+ import { MetaDataStore } from '../types';
7
+ import { convertMarkdownToHtml } from '../utils/convertMarkdownToHtml';
8
+ import { parseFrontmatter } from '../utils/frontmatter';
9
+
10
+ /**
11
+ * 生成语言切换器 HTML
12
+ * @param currentLang 当前语言
13
+ * @param availableLangs 可用语言列表
14
+ * @returns 语言切换器 HTML 字符串
15
+ */
16
+ function generateLanguageSwitcher(templateData: TemplateData): string {
17
+ const {
18
+ options: { langs = [] },
19
+ } = MetaData;
20
+
21
+ const items = langs
22
+ .map(lang => {
23
+ const langName = LANGUAGE_NAMES[lang] || lang;
24
+ const isCurrent = lang === templateData.lang;
25
+ const activeClass = isCurrent ? 'active' : '';
26
+
27
+ const link = path.join('..', lang, templateData.file.hash + '.html');
28
+
29
+ return `<li class="lang-item ${activeClass}">
30
+ <a href="${link}" class="lang-link">${langName}</a>
31
+ </li>`;
32
+ })
33
+ .join('');
34
+
35
+ return `<div class="language-switcher">
36
+ <ul class="lang-list">${items}</ul>
37
+ </div>`;
38
+ }
39
+
40
+ const generateTagsHtml = (tags: string[]): string => {
41
+ return `<ul class="tags-list">${tags
42
+ .map(tag => `<li class="tag-item">${tag}</li>`)
43
+ .join('')}</ul>`;
44
+ };
45
+
46
+ /**
47
+ * 生成导航 HTML
48
+ * @param navigation 导航树
49
+ * @param currentPath 当前路径(可选,用于高亮当前页面)
50
+ * @returns 导航 HTML 字符串
51
+ */
52
+ async function generateNavigationHtml(data: TemplateData): Promise<string> {
53
+ const { files } = MetaData;
54
+
55
+ const navigation = await Promise.all(
56
+ files.map(async file => {
57
+ const content = await fs.readFile(
58
+ path.join(ZEN_SRC_DIR, data.lang, file.hash + '.md'),
59
+ 'utf-8'
60
+ );
61
+ const { frontmatter } = parseFrontmatter(content);
62
+ const title = frontmatter.title || file.metadata?.title || file.path; // 优先使用提取的标题
63
+
64
+ // 使用相对链接
65
+ const link = file.hash + '.html';
66
+
67
+ return {
68
+ title,
69
+ link,
70
+ isActive: data.file.hash === file.hash,
71
+ };
72
+ })
73
+ );
74
+
75
+ return `<ul class="nav-list">${navigation
76
+ .map(item => {
77
+ const activeClass = item.isActive ? 'active' : '';
78
+
79
+ let html = `<li class="nav-item">`;
80
+ html += `<a href="${item.link}" class="nav-link ${activeClass}">${item.title}</a>`;
81
+
82
+ html += `</li>`;
83
+ return html;
84
+ })
85
+ .join('')}</ul>`;
86
+ }
87
+
88
+ const replaceInnerLinks = (data: TemplateData, markdownContent: string): string => {
89
+ let content = markdownContent;
90
+ for (const link of data.file.links) {
91
+ if (URL.canParse(link)) continue; // 跳过绝对 URL
92
+
93
+ const targetPath = path.resolve('/', path.dirname(data.file.path), link).slice(1);
94
+
95
+ const targetFile = MetaData.files.find(f => f.path === targetPath);
96
+
97
+ if (!targetFile) {
98
+ console.warn(`⚠️ Link target not found for ${link} in file ${data.file.path}`);
99
+ continue;
100
+ }
101
+ // 替换链接 (使用相对链接)
102
+ const targetLink = path.join(targetFile.hash + '.html');
103
+
104
+ // 全局替换链接
105
+ const linksRegex = new RegExp(`\\]\\(${link.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}\\)`, 'g');
106
+ content = content.replace(linksRegex, `](${targetLink})`);
107
+ }
108
+ return content;
109
+ };
110
+
111
+ interface TemplateData {
112
+ file: MetaDataStore['files'][0];
113
+ content: string;
114
+ lang: string;
115
+ }
116
+
117
+ /**
118
+ * 简单的模板变量替换
119
+ * @param template 模板字符串
120
+ * @param data 模板数据
121
+ * @returns 渲染后的 HTML 字符串
122
+ */
123
+ async function renderTemplate(template: string, data: TemplateData): Promise<string> {
124
+ const {
125
+ options: { langs = [] },
126
+ } = MetaData;
127
+ const markdownContent = data.content;
128
+ const { frontmatter, body } = parseFrontmatter(markdownContent);
129
+
130
+ const htmlContent = convertMarkdownToHtml(replaceInnerLinks(data, body));
131
+
132
+ let result = template;
133
+
134
+ // 替换导航
135
+ const navigationHtml = await generateNavigationHtml(data);
136
+ result = result.replace(/{{navigation}}/g, navigationHtml);
137
+
138
+ // 替换其他变量 - 使用全局替换
139
+ result = result.replace(/{{title}}/g, frontmatter.title || 'Untitled');
140
+ result = result.replace(/{{content}}/g, htmlContent);
141
+
142
+ // 替换元数据变量
143
+ if (frontmatter) {
144
+ result = result.replace(/{{summary}}/g, frontmatter.summary || '');
145
+ result = result.replace(/{{tags}}/g, generateTagsHtml(frontmatter.tags || []));
146
+ result = result.replace(/{{inferred_date}}/g, frontmatter.inferred_date || '--');
147
+ result = result.replace(/{{inferred_lang}}/g, frontmatter.inferred_lang || '--');
148
+ }
149
+
150
+ // 替换语言相关变量
151
+ result = result.replace(/{{lang}}/g, data.lang || '');
152
+ if (langs && langs.length > 1 && data.lang) {
153
+ const langSwitcher = generateLanguageSwitcher(data);
154
+ result = result.replace(/{{language_switcher}}/g, langSwitcher);
155
+ }
156
+
157
+ return result;
158
+ }
159
+
160
+ const renderRedirectTemplate = async (from: string, to: string): Promise<void> => {
161
+ const toURL = path.relative(path.dirname(from), to);
162
+ const html = `<!DOCTYPE html>
163
+ <html lang="en">
164
+ <head>
165
+ <meta charset="UTF-8">
166
+ <meta http-equiv="X-UA-Compatible" content="IE=edge">
167
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
168
+ <meta http-equiv="refresh" content="0; url=${toURL}">
169
+ <title>Redirecting...</title>
170
+ </head>
171
+ <body>
172
+ <p>Redirecting to <a href="${toURL}">${toURL}</a></p>
173
+ </body>
174
+ </html>`;
175
+ const targetPath = path.join(ZEN_DIST_DIR, from);
176
+ await fs.mkdir(path.dirname(targetPath), { recursive: true });
177
+ await fs.writeFile(targetPath, html, 'utf-8');
178
+ };
179
+
180
+ /**
181
+ * 渲染模板并保存文件
182
+ */
183
+ export async function renderTemplates(): Promise<void> {
184
+ const {
185
+ files,
186
+ options: { langs, verbose },
187
+ } = MetaData;
188
+
189
+ if (verbose) console.log(`⚡ Processing files...`);
190
+ const layoutTemplate = await fs.readFile(
191
+ path.join(__dirname, '../../assets/templates/default/layout.html'),
192
+ 'utf-8'
193
+ );
194
+
195
+ for (const file of files) {
196
+ for (const lang of langs || []) {
197
+ console.info(`📄 Preparing file for language: ${file.path} [${file.hash}] [${lang}]`);
198
+ const targetPath = path.join(ZEN_DIST_DIR, lang, file.hash + '.html');
199
+ const content = await fs.readFile(path.join(ZEN_SRC_DIR, lang, file.hash + '.md'), 'utf-8');
200
+ try {
201
+ const html = await renderTemplate(layoutTemplate, {
202
+ file,
203
+ content,
204
+ lang,
205
+ });
206
+ await fs.mkdir(path.dirname(targetPath), { recursive: true });
207
+ await fs.writeFile(targetPath, html, 'utf-8');
208
+ if (verbose) console.log(`✅ Rendered: ${targetPath}`);
209
+ } catch (error) {
210
+ console.error(`❌ Failed to render ${file.path}:`, error);
211
+ }
212
+ }
213
+ }
214
+
215
+ for (const lang of langs || []) {
216
+ await renderRedirectTemplate(
217
+ path.join(lang, 'index.html'),
218
+ path.join(lang, files[0].hash + '.html')
219
+ );
220
+ }
221
+ await renderRedirectTemplate('index.html', path.join(langs?.[0] || 'en-US', 'index.html'));
222
+ }
@@ -0,0 +1,92 @@
1
+ /**
2
+ * OpenAI 消息接口
3
+ */
4
+ export interface OpenAIMessage {
5
+ role: 'system' | 'user' | 'assistant';
6
+ content: string;
7
+ }
8
+
9
+ /**
10
+ * OpenAI 响应接口
11
+ */
12
+ export interface OpenAIResponse {
13
+ id: string;
14
+ object: string;
15
+ created: number;
16
+ model: string;
17
+ choices: Array<{
18
+ index: number;
19
+ message: {
20
+ role: string;
21
+ content: string;
22
+ };
23
+ finish_reason: string;
24
+ }>;
25
+ usage: {
26
+ prompt_tokens: number;
27
+ completion_tokens: number;
28
+ total_tokens: number;
29
+ };
30
+ }
31
+
32
+ /**
33
+ * 使用 OpenAI API 补全消息
34
+ * @param messages 消息数组
35
+ * @param options 可选配置
36
+ * @returns Promise<OpenAIResponse> 返回完整的OpenAI响应
37
+ */
38
+ export const completeMessages = async (
39
+ messages: OpenAIMessage[],
40
+ options?: {
41
+ response_format?: { type: 'json_object' | 'text' };
42
+ }
43
+ ): Promise<OpenAIResponse> => {
44
+ // 从环境变量读取配置
45
+ const apiKey = process.env.OPENAI_API_KEY || '';
46
+ const baseUrl = process.env.OPENAI_BASE_URL || 'https://api.openai.com/v1';
47
+ const model = process.env.OPENAI_MODEL || 'gpt-3.5-turbo';
48
+
49
+ if (!apiKey) {
50
+ throw new Error('OPENAI_API_KEY environment variable is not set');
51
+ }
52
+
53
+ try {
54
+ const requestBody: any = {
55
+ model,
56
+ messages,
57
+ temperature: 0, // 总是设置为 0,提取内容不需要随机性
58
+ // 不设置 max_tokens,让API自动决定
59
+ };
60
+
61
+ // 添加可选的response_format
62
+ if (options?.response_format) {
63
+ requestBody.response_format = options.response_format;
64
+ }
65
+
66
+ const response = await fetch(`${baseUrl}/chat/completions`, {
67
+ method: 'POST',
68
+ headers: {
69
+ 'Content-Type': 'application/json',
70
+ Authorization: `Bearer ${apiKey}`,
71
+ },
72
+ body: JSON.stringify(requestBody),
73
+ });
74
+
75
+ if (!response.ok) {
76
+ const errorText = await response.text();
77
+ throw new Error(`OpenAI API error (${response.status}): ${errorText}`);
78
+ }
79
+
80
+ const data: OpenAIResponse = await response.json();
81
+
82
+ // 验证响应
83
+ if (!data.choices?.[0]?.message?.content?.trim()) {
84
+ throw new Error('Empty response from OpenAI API');
85
+ }
86
+
87
+ return data;
88
+ } catch (error) {
89
+ console.error('❌ Failed to call OpenAI API:', error);
90
+ throw error;
91
+ }
92
+ };
package/src/types.ts ADDED
@@ -0,0 +1,37 @@
1
+ /**
2
+ * 单个文件的元数据缓存项
3
+ */
4
+ export interface FileMetaData {
5
+ hash: string;
6
+ path: string;
7
+ links: string[];
8
+ metadata?: AIMetadata;
9
+ }
10
+
11
+ /**
12
+ * .zen/meta.json 文件结构
13
+ */
14
+ export interface MetaDataStore {
15
+ version: string;
16
+ options: BuildOptions;
17
+ files: FileMetaData[];
18
+ }
19
+
20
+ export interface BuildOptions {
21
+ template?: string;
22
+ verbose?: boolean;
23
+ langs?: string[]; // 目标语言数组
24
+ }
25
+
26
+ export interface AIMetadata {
27
+ title: string; // AI 提取的标题
28
+ summary: string; // AI 提取的摘要,控制在 100字以内
29
+ tags: string[]; // AI 提取的关键字
30
+ inferred_date?: string; // 正文中隐含的文档创建日期,没有就留空
31
+ inferred_lang: string; // 文章使用的语言,例如 zh-Hans 或者 en-US
32
+ tokens_used?: {
33
+ prompt: number;
34
+ completion: number;
35
+ total: number;
36
+ }; // tokens 使用情况
37
+ }
@@ -0,0 +1,93 @@
1
+ import hljs from 'highlight.js';
2
+ import { marked } from 'marked';
3
+ import markedKatex from 'marked-katex-extension';
4
+
5
+ // 辅助函数:转义 HTML 特殊字符
6
+ function escapeHtml(unsafe: string): string {
7
+ return unsafe
8
+ .replace(/&/g, '&amp;')
9
+ .replace(/</g, '&lt;')
10
+ .replace(/>/g, '&gt;')
11
+ .replace(/"/g, '&quot;')
12
+ .replace(/'/g, '&#039;');
13
+ }
14
+ marked.use(markedKatex({ throwOnError: false, nonStandard: true } as any));
15
+
16
+ /**
17
+ * 将 Markdown 内容转换为 HTML
18
+ * @param mdContent Markdown 内容字符串
19
+ * @returns 转换后的 HTML 字符串
20
+ */
21
+ export const convertMarkdownToHtml = (mdContent: string): string => {
22
+ // 创建自定义渲染器
23
+ const renderer = new marked.Renderer();
24
+ const originalCodeRenderer = renderer.code;
25
+
26
+ // 重写代码块渲染器以支持 Mermaid - 使用 any 类型绕过类型检查
27
+ (renderer as any).code = function (code: any, language?: string, isEscaped?: boolean) {
28
+ // 在 marked 17+ 中,code 参数是一个对象,包含 text 和 lang 属性
29
+ const codeText = typeof code === 'string' ? code : code?.text || '';
30
+ // 语言信息在 code.lang 中,而不是 language 参数
31
+ const lang = code?.lang || language;
32
+
33
+ // 检测 Mermaid 代码块
34
+ if (lang === 'mermaid') {
35
+ // 生成唯一的 ID 用于图表容器
36
+ const chartId = 'mermaid-' + Date.now() + '-' + Math.random().toString(36).substr(2, 9);
37
+ return `
38
+ <div class="mermaid-diagram" data-mermaid-id="${chartId}">
39
+ <pre class="mermaid">${escapeHtml(codeText)}</pre>
40
+ </div>
41
+ `;
42
+ }
43
+
44
+ // 其他代码块使用原有高亮逻辑
45
+ return (originalCodeRenderer as any).call(this, code, language, isEscaped);
46
+ };
47
+
48
+ // 使用 marked.parse 的同步版本
49
+ // marked 17+ 默认返回 Promise,但我们可以使用 marked.parseSync 或 marked.parse 的同步模式
50
+ // 这里我们使用 marked.parse 并假设它是同步的(对于简单情况)
51
+ try {
52
+ // 尝试同步解析
53
+ const result = marked.parse(mdContent, {
54
+ renderer,
55
+ highlight: function (code: string, lang: string) {
56
+ // 跳过 Mermaid 代码块的高亮
57
+ if (lang === 'mermaid') {
58
+ return code;
59
+ }
60
+
61
+ if (lang && hljs.getLanguage(lang)) {
62
+ try {
63
+ return hljs.highlight(code, { language: lang }).value;
64
+ } catch (err) {
65
+ console.warn(`Failed to highlight code with language ${lang}:`, err);
66
+ }
67
+ }
68
+ return hljs.highlightAuto(code).value;
69
+ },
70
+ pedantic: false,
71
+ gfm: true,
72
+ breaks: false,
73
+ sanitize: false,
74
+ smartLists: true,
75
+ smartypants: false,
76
+ xhtml: false,
77
+ async: false, // 强制同步模式
78
+ } as any);
79
+
80
+ // 如果结果是 Promise,等待它(虽然我们设置了 async: false)
81
+ if (result && typeof result.then === 'function') {
82
+ // 这不应该发生,但如果发生了,返回一个占位符
83
+ console.warn('marked.parse returned a Promise despite async: false');
84
+ return '<!-- Markdown conversion in progress -->';
85
+ }
86
+
87
+ return result as unknown as string;
88
+ } catch (error) {
89
+ console.error('Error converting Markdown to HTML:', error);
90
+ const errorMessage = error instanceof Error ? error.message : String(error);
91
+ return `<div class="error">Error converting Markdown: ${errorMessage}</div>`;
92
+ }
93
+ };
@@ -0,0 +1,18 @@
1
+ import { parse, stringify } from 'yaml';
2
+
3
+ export const parseFrontmatter = (content: string): { frontmatter: any; body: string } => {
4
+ const frontmatterRegex = /^---\n([\s\S]*?)\n---/;
5
+ const match = content.match(frontmatterRegex);
6
+ if (match) {
7
+ const frontmatterContent = match[1];
8
+ const body = content.slice(match[0].length).trim();
9
+ return { frontmatter: parse(frontmatterContent.trim()), body };
10
+ }
11
+ return { frontmatter: {}, body: content };
12
+ };
13
+
14
+ export const updateFrontmatter = (content: string, newFrontmatter: any): string => {
15
+ const { body } = parseFrontmatter(content);
16
+ const frontmatterContent = `---\n${stringify(newFrontmatter, { defaultStringType: 'QUOTE_DOUBLE' })}---\n\n`;
17
+ return frontmatterContent + body;
18
+ };
@@ -0,0 +1,4 @@
1
+ import { createHash } from 'crypto';
2
+
3
+ export const sha256 = (content: string): string =>
4
+ createHash('sha256').update(content).digest('hex');