czon 0.1.1 → 0.1.3
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/.github/workflows/pages.yml +1 -1
- package/README.md +7 -7
- package/assets/templates/default/layout.html +2 -6
- package/dist/build/pipeline.d.ts.map +1 -1
- package/dist/build/pipeline.js +11 -11
- package/dist/build/pipeline.js.map +1 -1
- package/dist/cli.js +5 -5
- package/dist/cli.js.map +1 -1
- package/dist/findEntries.d.ts +1 -1
- package/dist/findEntries.js +2 -2
- package/dist/findEntries.js.map +1 -1
- package/dist/metadata.d.ts.map +1 -1
- package/dist/metadata.js +4 -7
- package/dist/metadata.js.map +1 -1
- package/dist/paths.d.ts +4 -4
- package/dist/paths.d.ts.map +1 -1
- package/dist/paths.js +5 -5
- package/dist/paths.js.map +1 -1
- package/dist/process/template.js +4 -4
- package/dist/process/template.js.map +1 -1
- package/dist/types.d.ts +1 -1
- package/package.json +1 -1
- package/.zen/meta.json +0 -106
- package/.zen/src/ar-SA/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/ar-SA/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/ar-SA/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/ar-SA/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/ar-SA/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/ar-SA/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/da-DK/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/da-DK/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/da-DK/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/da-DK/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/da-DK/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/da-DK/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/de-DE/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/de-DE/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/de-DE/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/de-DE/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/de-DE/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/de-DE/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/en-US/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/en-US/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/en-US/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/en-US/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/en-US/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/en-US/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/es-ES/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/es-ES/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/es-ES/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/es-ES/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/es-ES/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/es-ES/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/es-MX/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/es-MX/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/es-MX/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/es-MX/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/es-MX/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/es-MX/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/fi-FI/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/fi-FI/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/fi-FI/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/fi-FI/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/fi-FI/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/fi-FI/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/fr-FR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/fr-FR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/fr-FR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/fr-FR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/fr-FR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/fr-FR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/hi-IN/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/hi-IN/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/hi-IN/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/hi-IN/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/hi-IN/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/hi-IN/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/id-ID/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/id-ID/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/id-ID/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/id-ID/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/id-ID/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/id-ID/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/it-IT/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/it-IT/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/it-IT/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/it-IT/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/it-IT/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/it-IT/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/ja-JP/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/ja-JP/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/ja-JP/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/ja-JP/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/ja-JP/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/ja-JP/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/ko-KR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/ko-KR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/ko-KR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/ko-KR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/ko-KR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/ko-KR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/nl-NL/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/nl-NL/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/nl-NL/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/nl-NL/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/nl-NL/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/nl-NL/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/no-NO/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/no-NO/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/no-NO/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/no-NO/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/no-NO/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/no-NO/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/pl-PL/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/pl-PL/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/pl-PL/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/pl-PL/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/pl-PL/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/pl-PL/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/pt-BR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/pt-BR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/pt-BR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/pt-BR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/pt-BR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/pt-BR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/pt-PT/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/pt-PT/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/pt-PT/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/pt-PT/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/pt-PT/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/pt-PT/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/ru-RU/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/ru-RU/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/ru-RU/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/ru-RU/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/ru-RU/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/ru-RU/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/sv-SE/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/sv-SE/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/sv-SE/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/sv-SE/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/sv-SE/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/sv-SE/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/th-TH/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/th-TH/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/th-TH/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/th-TH/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/th-TH/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/th-TH/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/tr-TR/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/tr-TR/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/tr-TR/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/tr-TR/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/tr-TR/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/tr-TR/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/uk-UA/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/uk-UA/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -79
- package/.zen/src/uk-UA/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/uk-UA/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/uk-UA/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/uk-UA/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/vi-VN/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -121
- package/.zen/src/vi-VN/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -78
- package/.zen/src/vi-VN/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/vi-VN/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -71
- package/.zen/src/vi-VN/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/vi-VN/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -30
- package/.zen/src/zh-Hans/4b2cedd76dddb2a20b1890460735dd38fa65e293a37d6bd15310852ba1c851c4.md +0 -21
- package/.zen/src/zh-Hans/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -123
- package/.zen/src/zh-Hans/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -81
- package/.zen/src/zh-Hans/59fe7aff1dcf6f4bbbaff1ad2f7c4c5a3d34b99b1d288b78717ee34525c9897b.md +0 -24
- package/.zen/src/zh-Hans/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -74
- package/.zen/src/zh-Hans/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -74
- package/.zen/src/zh-Hans/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -75
- package/.zen/src/zh-Hans/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -31
- package/.zen/src/zh-Hans/c1b0e9eb01d78140c21097187d6395815fd01a6f179af31b7e72532ec6c7c682.md +0 -75
- package/.zen/src/zh-Hans/cbf2fcda767ee28fe49828908c14de2c8edba8f8b7d721743eda767838b70923.md +0 -67
- package/.zen/src/zh-Hans/ec9fa686ad66e28d8d9ee131812bb6f31052185584de0853db1ee8893ebddda3.md +0 -22
- package/.zen/src/zh-Hans/f119cbaf7645c713b1e0fa67abdc1070766c80f94eec9e42108bfa37ef85db43.md +0 -24
- package/.zen/src/zh-Hant/50bf41ac6fd8ec8cd6481fd1114aaf00abed0461fc8e8c81cc0373d38ffa835c.md +0 -123
- package/.zen/src/zh-Hant/579c3a203225b8b7aead541f42381de0619d8bc1d451d0d9c3267dbc74205503.md +0 -81
- package/.zen/src/zh-Hant/6e7ca196ba51235db30218cf9b28f92d35937b09a22411062be088a0086b29ed.md +0 -70
- package/.zen/src/zh-Hant/74541be9d53c64107a548e09847244d52bff803960942cd2bfbd2a8600afb805.md +0 -74
- package/.zen/src/zh-Hant/98a9dd72146a69d155030cb7286ea2d10e0521451de9754cbdfcde46c46f2c71.md +0 -71
- package/.zen/src/zh-Hant/9d0485c244a80b14dcc92ef91cf5ff82220ed774ebb5e09549929ac75200c7bc.md +0 -31
- package/docs/test-katex.md +0 -66
- package/docs/test-mermaid.md +0 -105
- package/docs/tricks.md +0 -15
- package/src/ai/extractMetadataFromMarkdown.ts +0 -95
- package/src/ai/translateMarkdown.ts +0 -60
- package/src/build/pipeline.ts +0 -172
- package/src/cli.ts +0 -73
- package/src/findEntries.ts +0 -37
- package/src/index.ts +0 -1
- package/src/languages.ts +0 -37
- package/src/metadata.ts +0 -44
- package/src/paths.ts +0 -7
- package/src/process/extractMetadataByAI.ts +0 -31
- package/src/process/scanSourceFiles.ts +0 -73
- package/src/process/template.ts +0 -222
- package/src/services/openai.ts +0 -92
- package/src/types.ts +0 -37
- package/src/utils/convertMarkdownToHtml.ts +0 -93
- package/src/utils/frontmatter.ts +0 -18
- package/src/utils/sha256.ts +0 -4
- package/test-multilang.js +0 -44
|
@@ -1,95 +0,0 @@
|
|
|
1
|
-
import { completeMessages, OpenAIMessage } from '../services/openai';
|
|
2
|
-
import { AIMetadata } from '../types';
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
* 从 markdown 内容中提取 metadata
|
|
6
|
-
* @param content Markdown 内容
|
|
7
|
-
* @returns Promise<AIMetadata> 提取的元数据,失败时抛出错误
|
|
8
|
-
*/
|
|
9
|
-
export async function extractMetadataFromMarkdown(content: string): Promise<AIMetadata> {
|
|
10
|
-
const prompt = buildMetadataPrompt(content);
|
|
11
|
-
const messages: OpenAIMessage[] = [
|
|
12
|
-
{
|
|
13
|
-
role: 'system',
|
|
14
|
-
content:
|
|
15
|
-
'你是一个专业的文档分析助手,擅长从文档中提取结构化信息。请严格按照要求的 JSON 格式返回结果。',
|
|
16
|
-
},
|
|
17
|
-
{
|
|
18
|
-
role: 'user',
|
|
19
|
-
content: prompt,
|
|
20
|
-
},
|
|
21
|
-
];
|
|
22
|
-
|
|
23
|
-
const response = await completeMessages(messages, {
|
|
24
|
-
response_format: { type: 'json_object' },
|
|
25
|
-
});
|
|
26
|
-
|
|
27
|
-
const metadata = parseMetadataResponse(response.choices[0].message.content);
|
|
28
|
-
|
|
29
|
-
// 添加 tokens 使用情况
|
|
30
|
-
metadata.tokens_used = {
|
|
31
|
-
prompt: response.usage.prompt_tokens,
|
|
32
|
-
completion: response.usage.completion_tokens,
|
|
33
|
-
total: response.usage.total_tokens,
|
|
34
|
-
};
|
|
35
|
-
|
|
36
|
-
return metadata;
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
/**
|
|
40
|
-
* 构建提取 metadata 的 prompt
|
|
41
|
-
*/
|
|
42
|
-
function buildMetadataPrompt(content: string): string {
|
|
43
|
-
// 限制内容长度以避免 token 超限
|
|
44
|
-
const maxContentLength = Infinity; // 可根据需要调整长度限制
|
|
45
|
-
const truncatedContent =
|
|
46
|
-
content.length > maxContentLength
|
|
47
|
-
? content.substring(0, maxContentLength) + '... [内容已截断]'
|
|
48
|
-
: content;
|
|
49
|
-
|
|
50
|
-
return `请分析以下文档内容,提取以下信息并返回 JSON 格式:
|
|
51
|
-
|
|
52
|
-
文档内容:
|
|
53
|
-
"""
|
|
54
|
-
${truncatedContent}
|
|
55
|
-
"""
|
|
56
|
-
|
|
57
|
-
请提取:
|
|
58
|
-
1. title: 文档的标题(简洁明了,不超过 30 个字)
|
|
59
|
-
2. summary: 文档摘要(控制在 300 字以内,概括主要内容)
|
|
60
|
-
3. tags: 关键词列表(3-8 个关键词,使用中文或英文)
|
|
61
|
-
4. inferred_date: 文档中隐含的创建日期(如果有的话,格式:YYYY-MM-DD,没有就留空字符串)
|
|
62
|
-
5. inferred_lang: 文档使用的语言代码(例如:zh-Hans 表示简体中文,en-US 表示美式英语)
|
|
63
|
-
|
|
64
|
-
请严格按照以下 JSON 格式返回,不要包含任何其他文本:
|
|
65
|
-
{
|
|
66
|
-
"title": "文档标题",
|
|
67
|
-
"summary": "文档摘要...",
|
|
68
|
-
"tags": ["关键词1", "关键词2", "关键词3"],
|
|
69
|
-
"inferred_date": "2023-01-01",
|
|
70
|
-
"inferred_lang": "zh-Hans"
|
|
71
|
-
}`;
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
/**
|
|
75
|
-
* 解析 AI 返回的 metadata
|
|
76
|
-
*/
|
|
77
|
-
function parseMetadataResponse(responseContent: string): AIMetadata {
|
|
78
|
-
try {
|
|
79
|
-
const metadata = JSON.parse(responseContent);
|
|
80
|
-
|
|
81
|
-
// 验证和清理数据
|
|
82
|
-
return {
|
|
83
|
-
title: metadata.title?.trim() || '未命名文档',
|
|
84
|
-
summary: metadata.summary?.trim() || '',
|
|
85
|
-
tags: Array.isArray(metadata.tags)
|
|
86
|
-
? metadata.tags.map((tag: string) => tag.trim()).filter(Boolean)
|
|
87
|
-
: [],
|
|
88
|
-
inferred_date: metadata.inferred_date?.trim() || undefined,
|
|
89
|
-
inferred_lang: metadata.inferred_lang?.trim() || 'zh-Hans',
|
|
90
|
-
};
|
|
91
|
-
} catch (error) {
|
|
92
|
-
console.error('❌ Failed to parse AI response:', error, 'Response:', responseContent);
|
|
93
|
-
throw error;
|
|
94
|
-
}
|
|
95
|
-
}
|
|
@@ -1,60 +0,0 @@
|
|
|
1
|
-
import { LANGUAGE_NAMES } from '../languages';
|
|
2
|
-
import { completeMessages, OpenAIMessage } from '../services/openai';
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
|
|
6
|
-
* @param content Markdown 内容
|
|
7
|
-
* @param targetLang 目标语言代码(例如:zh-Hans, en-US)
|
|
8
|
-
* @returns Promise<string> 翻译后的 Markdown 内容
|
|
9
|
-
*/
|
|
10
|
-
export async function translateMarkdown(content: string, targetLang: string): Promise<string> {
|
|
11
|
-
const langName = LANGUAGE_NAMES[targetLang];
|
|
12
|
-
const lang = `${langName} (${targetLang})`;
|
|
13
|
-
const messages: OpenAIMessage[] = [
|
|
14
|
-
{
|
|
15
|
-
role: 'system',
|
|
16
|
-
content: [
|
|
17
|
-
`You are a professional document translator.`,
|
|
18
|
-
`Translate the following markdown content into ${lang}`,
|
|
19
|
-
targetLang === 'ja-JP'
|
|
20
|
-
? [
|
|
21
|
-
//
|
|
22
|
-
'使用日语母语者自然的表达方式。',
|
|
23
|
-
'采用适合技术/专业文档的礼貌、正式语体(丁寧体/です・ます体)',
|
|
24
|
-
'确保语法和字符集完全符合日语规范。',
|
|
25
|
-
'绝对禁止使用繁体中文汉字。',
|
|
26
|
-
'所有日语汉字必须使用标准的 **日本常用汉字(Jōyō kanji)** 字形。',
|
|
27
|
-
'特别检查以下字形示例,确保使用日文标准字形:',
|
|
28
|
-
'1. “国” - 使用“国”而非“國”',
|
|
29
|
-
'2. “学” - 使用“学”而非“學”',
|
|
30
|
-
'3. “広” - 使用“広”而非“廣”',
|
|
31
|
-
'4. “円” - 使用“円”而非“圓”',
|
|
32
|
-
'5. “医” - 使用“医”而非“醫”',
|
|
33
|
-
'6. “図” - 使用“図”而非“圖”',
|
|
34
|
-
'7. “対” - 使用“対”而非“對”',
|
|
35
|
-
'8. “声” - 使用“声”而非“聲”',
|
|
36
|
-
'9. “芸” - 使用“芸”而非“藝”',
|
|
37
|
-
'10. “験” - 使用“験”而非“驗”',
|
|
38
|
-
].join('\n')
|
|
39
|
-
: ``,
|
|
40
|
-
`Preserve the original markdown formatting, including headings, lists, code blocks, links, and images.`,
|
|
41
|
-
`Do not change any non-text elements or their formatting.`,
|
|
42
|
-
`Ensure that technical terms and code snippets remain unchanged.`,
|
|
43
|
-
`Provide a natural and fluent translation suitable for readers familiar with the subject matter.`,
|
|
44
|
-
].join('\n'),
|
|
45
|
-
},
|
|
46
|
-
{
|
|
47
|
-
role: 'user',
|
|
48
|
-
content: content,
|
|
49
|
-
},
|
|
50
|
-
];
|
|
51
|
-
|
|
52
|
-
const response = await completeMessages(messages);
|
|
53
|
-
const translatedContent = response.choices[0]?.message?.content?.trim() || '';
|
|
54
|
-
|
|
55
|
-
if (!translatedContent) {
|
|
56
|
-
throw new Error('Empty translation response');
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
return translatedContent;
|
|
60
|
-
}
|
package/src/build/pipeline.ts
DELETED
|
@@ -1,172 +0,0 @@
|
|
|
1
|
-
import * as fs from 'fs/promises';
|
|
2
|
-
import * as path from 'path';
|
|
3
|
-
import { translateMarkdown } from '../ai/translateMarkdown';
|
|
4
|
-
import { loadMetaData, MetaData, saveMetaData } from '../metadata';
|
|
5
|
-
import { INPUT_DIR, ZEN_DIR, ZEN_DIST_DIR, ZEN_SRC_DIR } from '../paths';
|
|
6
|
-
import { extractMetadataByAI } from '../process/extractMetadataByAI';
|
|
7
|
-
import { scanSourceFiles } from '../process/scanSourceFiles';
|
|
8
|
-
import { renderTemplates } from '../process/template';
|
|
9
|
-
import { BuildOptions } from '../types';
|
|
10
|
-
import { updateFrontmatter } from '../utils/frontmatter';
|
|
11
|
-
|
|
12
|
-
/**
|
|
13
|
-
* 验证构建配置
|
|
14
|
-
*/
|
|
15
|
-
async function validateConfig(options: BuildOptions): Promise<void> {
|
|
16
|
-
const { verbose = false } = options;
|
|
17
|
-
|
|
18
|
-
if (verbose) {
|
|
19
|
-
console.log(`🚀 Starting ZEN build...`);
|
|
20
|
-
if (options.langs && options.langs.length > 0) {
|
|
21
|
-
console.log(`🌐 Target languages: ${options.langs.join(', ')}`);
|
|
22
|
-
}
|
|
23
|
-
console.log(`🔍 Verbose mode enabled`);
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
MetaData.options = options;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
/**
|
|
30
|
-
* 存储母语文件到 .zen/src
|
|
31
|
-
*/
|
|
32
|
-
async function storeNativeFiles(): Promise<void> {
|
|
33
|
-
const {
|
|
34
|
-
options: { verbose },
|
|
35
|
-
files,
|
|
36
|
-
} = MetaData;
|
|
37
|
-
for (const file of MetaData.files) {
|
|
38
|
-
try {
|
|
39
|
-
if (!file.hash) throw new Error(`Missing hash`);
|
|
40
|
-
if (!file.metadata?.inferred_lang) throw new Error(`Missing inferred language`);
|
|
41
|
-
const filePath = path.join(ZEN_SRC_DIR, file.metadata.inferred_lang, file.hash + '.md');
|
|
42
|
-
const originalContent = await fs.readFile(path.join(INPUT_DIR, file.path), 'utf-8');
|
|
43
|
-
|
|
44
|
-
const enhancedContent = updateFrontmatter(originalContent, {
|
|
45
|
-
title: file.metadata.title,
|
|
46
|
-
summary: file.metadata.summary,
|
|
47
|
-
tags: file.metadata.tags,
|
|
48
|
-
inferred_date: file.metadata.inferred_date,
|
|
49
|
-
inferred_lang: file.metadata.inferred_lang,
|
|
50
|
-
});
|
|
51
|
-
await fs.mkdir(path.dirname(filePath), { recursive: true });
|
|
52
|
-
|
|
53
|
-
await fs.writeFile(filePath, enhancedContent, 'utf-8');
|
|
54
|
-
} catch (error) {
|
|
55
|
-
console.warn(`⚠️ Failed to store native file ${file.path}:`, error);
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
if (verbose && files.length > 0) {
|
|
60
|
-
console.log(`💾 Stored ${files.length} native language files to .zen/src`);
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
/**
|
|
65
|
-
* 处理翻译
|
|
66
|
-
*/
|
|
67
|
-
async function processTranslations(): Promise<void> {
|
|
68
|
-
const {
|
|
69
|
-
files,
|
|
70
|
-
options: { langs = [], verbose },
|
|
71
|
-
} = MetaData;
|
|
72
|
-
|
|
73
|
-
await Promise.all(
|
|
74
|
-
files.flatMap(async file => {
|
|
75
|
-
return Promise.all(
|
|
76
|
-
langs.map(async lang => {
|
|
77
|
-
if (verbose) console.info(`📄 Processing file for translation: ${file.path}`);
|
|
78
|
-
if (!file.metadata) {
|
|
79
|
-
console.warn(`⚠️ Missing metadata for file: ${file.path}, skipping translation.`);
|
|
80
|
-
return;
|
|
81
|
-
}
|
|
82
|
-
if (verbose) console.log(`🌐 Translating to ${lang}...`);
|
|
83
|
-
// 存储翻译文件到 .zen/src/{lang}
|
|
84
|
-
const sourcePath = path.join(ZEN_SRC_DIR, file.metadata.inferred_lang, file.hash + '.md'); // 使用已经加强的母语文件路径
|
|
85
|
-
const targetPath = path.join(ZEN_SRC_DIR, lang, file.hash + '.md');
|
|
86
|
-
|
|
87
|
-
try {
|
|
88
|
-
const content = await fs.readFile(sourcePath, 'utf-8');
|
|
89
|
-
if (file.metadata.inferred_lang === lang) {
|
|
90
|
-
if (verbose)
|
|
91
|
-
console.log(`ℹ️ Skipping translation for ${file.path}, already in target language`);
|
|
92
|
-
return;
|
|
93
|
-
} else {
|
|
94
|
-
// 翻译
|
|
95
|
-
// 先检查是否已经有翻译文件存在
|
|
96
|
-
|
|
97
|
-
const exists = await fs.access(targetPath).then(
|
|
98
|
-
() => true,
|
|
99
|
-
() => false
|
|
100
|
-
);
|
|
101
|
-
if (exists) {
|
|
102
|
-
if (verbose)
|
|
103
|
-
console.log(`ℹ️ Translation already exists for ${file.path} in ${lang}`);
|
|
104
|
-
return;
|
|
105
|
-
}
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
const translatedContent = await translateMarkdown(content, lang);
|
|
109
|
-
|
|
110
|
-
await fs.mkdir(path.dirname(targetPath), { recursive: true });
|
|
111
|
-
await fs.writeFile(targetPath, translatedContent, 'utf-8');
|
|
112
|
-
|
|
113
|
-
if (verbose) console.log(`✅ Translated file saved: ${targetPath}`);
|
|
114
|
-
} catch (error) {
|
|
115
|
-
console.error(`❌ Failed to translate to ${lang}:`, error);
|
|
116
|
-
}
|
|
117
|
-
})
|
|
118
|
-
);
|
|
119
|
-
})
|
|
120
|
-
);
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
/**
|
|
124
|
-
* 构建管道(函数组合)
|
|
125
|
-
*/
|
|
126
|
-
async function buildPipeline(options: BuildOptions): Promise<void> {
|
|
127
|
-
// 验证配置
|
|
128
|
-
await validateConfig(options);
|
|
129
|
-
|
|
130
|
-
// 清理输出目录
|
|
131
|
-
await fs.rm(ZEN_DIST_DIR, { recursive: true, force: true });
|
|
132
|
-
|
|
133
|
-
// 确保 .zen/.gitignore 文件
|
|
134
|
-
await fs.mkdir(ZEN_DIR, { recursive: true });
|
|
135
|
-
await fs.writeFile(path.join(ZEN_DIR, '.gitignore'), 'dist\n', 'utf-8');
|
|
136
|
-
|
|
137
|
-
// 扫描源文件
|
|
138
|
-
await scanSourceFiles();
|
|
139
|
-
|
|
140
|
-
// 运行 AI 元数据提取
|
|
141
|
-
await extractMetadataByAI();
|
|
142
|
-
|
|
143
|
-
// 存储母语文件
|
|
144
|
-
await storeNativeFiles();
|
|
145
|
-
|
|
146
|
-
// 处理翻译
|
|
147
|
-
await processTranslations();
|
|
148
|
-
|
|
149
|
-
// 渲染模板
|
|
150
|
-
await renderTemplates();
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
/**
|
|
154
|
-
* 主构建函数
|
|
155
|
-
*/
|
|
156
|
-
export async function buildSite(options: BuildOptions): Promise<void> {
|
|
157
|
-
const startTime = Date.now();
|
|
158
|
-
|
|
159
|
-
try {
|
|
160
|
-
await loadMetaData();
|
|
161
|
-
await buildPipeline(options);
|
|
162
|
-
|
|
163
|
-
const endTime = Date.now();
|
|
164
|
-
const duration = ((endTime - startTime) / 1000).toFixed(2);
|
|
165
|
-
console.log(`🎉 Build completed in ${duration}s`);
|
|
166
|
-
} catch (error) {
|
|
167
|
-
console.error(`❌ Build failed:`, error);
|
|
168
|
-
throw error;
|
|
169
|
-
} finally {
|
|
170
|
-
await saveMetaData();
|
|
171
|
-
}
|
|
172
|
-
}
|
package/src/cli.ts
DELETED
|
@@ -1,73 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
|
|
3
|
-
import { Cli, Command, Option } from 'clipanion';
|
|
4
|
-
import { config } from 'dotenv';
|
|
5
|
-
import * as fs from 'fs';
|
|
6
|
-
import * as path from 'path';
|
|
7
|
-
import { buildSite } from './build/pipeline';
|
|
8
|
-
|
|
9
|
-
// 加载 .env 文件中的环境变量
|
|
10
|
-
config();
|
|
11
|
-
|
|
12
|
-
// 获取版本号 - 从 package.json 读取
|
|
13
|
-
function getVersion(): string {
|
|
14
|
-
const packageJsonPath = path.join(__dirname, '..', 'package.json');
|
|
15
|
-
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
|
|
16
|
-
return packageJson.version;
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
// Build 命令
|
|
20
|
-
class BuildCommand extends Command {
|
|
21
|
-
static paths = [['build']];
|
|
22
|
-
|
|
23
|
-
template = Option.String('-t,--template');
|
|
24
|
-
verbose = Option.Boolean('-v,--verbose');
|
|
25
|
-
lang = Option.Array('--lang', {
|
|
26
|
-
description: 'Target languages for translation (e.g., en-US, ja-JP)',
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
static usage = Command.Usage({
|
|
30
|
-
description: 'Build documentation site from Markdown files in current directory',
|
|
31
|
-
details: `
|
|
32
|
-
This command builds a documentation site from Markdown files in the current directory.
|
|
33
|
-
The output will be placed in the .zen/dist directory.
|
|
34
|
-
|
|
35
|
-
Examples:
|
|
36
|
-
$ zengen build
|
|
37
|
-
$ zengen build --lang en-US --lang ja-JP (translate to English and Japanese)
|
|
38
|
-
`,
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
async execute() {
|
|
42
|
-
try {
|
|
43
|
-
await buildSite({
|
|
44
|
-
template: this.template ? path.resolve(this.template) : undefined,
|
|
45
|
-
verbose: this.verbose,
|
|
46
|
-
langs: this.lang,
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
return 0;
|
|
50
|
-
} catch (error) {
|
|
51
|
-
this.context.stderr.write(`❌ Build failed: ${error}\n`);
|
|
52
|
-
return 1;
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
// 创建 CLI 应用
|
|
58
|
-
const cli = new Cli({
|
|
59
|
-
binaryName: 'zengen',
|
|
60
|
-
binaryLabel: 'ZEN - A minimalist Markdown documentation site builder',
|
|
61
|
-
binaryVersion: getVersion(),
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
// 注册命令
|
|
65
|
-
cli.register(BuildCommand);
|
|
66
|
-
|
|
67
|
-
// 运行 CLI
|
|
68
|
-
cli.runExit(process.argv.slice(2), {
|
|
69
|
-
...Cli.defaultContext,
|
|
70
|
-
stdin: process.stdin,
|
|
71
|
-
stdout: process.stdout,
|
|
72
|
-
stderr: process.stderr,
|
|
73
|
-
});
|
package/src/findEntries.ts
DELETED
|
@@ -1,37 +0,0 @@
|
|
|
1
|
-
import { exec } from 'child_process';
|
|
2
|
-
import { promisify } from 'util';
|
|
3
|
-
|
|
4
|
-
const execAsync = promisify(exec);
|
|
5
|
-
|
|
6
|
-
/**
|
|
7
|
-
* 使用git命令查找所有Markdown文件
|
|
8
|
-
* 使用git ls-files --others --cached --exclude-standard获取所有文件
|
|
9
|
-
* 然后过滤掉.zen目录和只保留.md文件
|
|
10
|
-
*
|
|
11
|
-
* @param dirPath 要扫描的目录路径
|
|
12
|
-
* @returns Promise<string[]> 返回Markdown文件的相对路径数组
|
|
13
|
-
*/
|
|
14
|
-
export const findMarkdownEntries = async (dirPath: string): Promise<string[]> => {
|
|
15
|
-
try {
|
|
16
|
-
// 使用git命令获取所有文件(包括已跟踪和未跟踪的文件)
|
|
17
|
-
// 在指定的目录下执行git命令
|
|
18
|
-
const { stdout } = await execAsync('git ls-files --others --cached --exclude-standard', {
|
|
19
|
-
cwd: dirPath,
|
|
20
|
-
});
|
|
21
|
-
|
|
22
|
-
// 按行分割并过滤
|
|
23
|
-
const files = stdout
|
|
24
|
-
.split('\n')
|
|
25
|
-
.filter(line => line.trim() !== '') // 移除空行
|
|
26
|
-
.filter(file => !file.startsWith('.zen')) // 过滤掉.zen目录下的文件
|
|
27
|
-
.filter(file => file.endsWith('.md')); // 只保留.md文件
|
|
28
|
-
|
|
29
|
-
return files;
|
|
30
|
-
} catch (error) {
|
|
31
|
-
console.error('Error finding markdown entries:', error);
|
|
32
|
-
|
|
33
|
-
// 如果git命令失败,返回空数组
|
|
34
|
-
// 这可以处理没有git仓库或git不可用的情况
|
|
35
|
-
return [];
|
|
36
|
-
}
|
|
37
|
-
};
|
package/src/index.ts
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
import './cli';
|
package/src/languages.ts
DELETED
|
@@ -1,37 +0,0 @@
|
|
|
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
|
-
};
|
package/src/metadata.ts
DELETED
|
@@ -1,44 +0,0 @@
|
|
|
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
DELETED
|
@@ -1,7 +0,0 @@
|
|
|
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');
|
|
@@ -1,31 +0,0 @@
|
|
|
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
|
-
}
|
|
@@ -1,73 +0,0 @@
|
|
|
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
|
-
}
|