I revived an archived project and released it as FortuneExcel - a plugin for FortuneSheet to import/export Excel (.xlsx) files.
Initialization
Reviving the project meant fixing the deprecated code and existing bugs. Along with this, I added export feature, toolbar plugin support for fortune-sheet and set up Github workflow.
Modularization
While implementing a package, it should have maximum abstraction. User should have to add minimal code to their environment for your package to work, all while having no side-effects on their existing code. A proper code strucrure also makes it easier for others to contribute to your code.
Github Workflow
Lets breakdown the pubhlish.yml file.
Run the workflow whenever a release is published or manually triggered from Actions tab:
publish.yml
-----------
on:
release:
types: [published]
workflow_dispatch:
Define premissions required by the workflow:
permissions:
contents: read
pages: write
id-token: write
Checkout the code, setup node environment, install dependencies (see npm i vs npm ci) and publish the package on npm:
- uses: actions/checkout@v3
- uses: actions/setup-node@v3
with:
node-version: "20.x"
registry-url: "https://registry.npmjs.org"
- run: npm ci
- run: npm publish --tag latest --provenance --access public
env:
NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }}
Build the Storybook, upload it as an artifact and deploy it on GitHub Pages:
Note: Update Build and Deployement source to ‘Github Actions’ in repo > Settings > Pages
- run: npm run build-storybook
- uses: actions/upload-pages-artifact@v2
with:
path: storybook-static
- id: deploy
uses: actions/deploy-pages@v3
with:
token: ${{ github.token }}
Secrets and Tokens
In the above section inside publish.yml, notice github.token
and secrets.NPM_TOKEN
.
While github.token
is automatically provided by GitHub Actions, secrets.NPM_TOKEN
is an automation access token that has to be generated at npmjs.com and set as repository secret on Github.
Emitted Javascript version
Fortune-excel and fortune-sheet were having incompatibility issues due to different transmitted JS when compiled from TS. This is defined by the target
value in compilerOptions
of tsconfig.json. Since fortune-sheet compiles the JS In es5, changing the target from es6 to es5 in fortune-excel fixed the issue:
// tsconfig.json
{
"compilerOptions": {
"target": "es5",
...
}
}
Commonjs entrypoint
After publishing the package, Yarn showed this warning - ‘the package doesn’t seem to have a commonjs entry point’. Compiling the TS in commonjs is necessary for old projects still using CJS instead of ESM. Compiling is ESM is optional since EMS supports CJS, but not visa-versa. To fix the warning:
- Make sure package.json does not have
"type": "module"
definedWhen set to “module”, the type field allows a package to specify all .js files within are ES modules. If the “type” field is omitted or set to “commonjs”, all .js files are treated as CommonJS.
- Define
"module": "commonjs"
in tsconfig.json - refer to this.
Compiling TS for Storybook
To make our plugin work with Storybook in the same repository, we can take two approaches:
- Using Compiled JavaScript (CommonJS from
/dist/
) - Using TypeScript Source Directly
Using TS source directly enables seamless debugging in browsers, but it will require additional configuration. To leverage your TypeScript source directly within Storybook:
Configure TypeScript Path Aliases: Modify your tsconfig.json to create a path alias for your plugin’s entry point. This allows Storybook to resolve the import correctly.
// tsconfig.json { "paths": { "@corbe30/fortune-excel": ["./src/main.ts"], }, ... }
// stories/Page.tsx import { exportToolBarItem, importToolBarItem, } from "@corbe30/fortune-excel";
Configure Babel for TypeScript Support: Storybook uses Babel for its build process. To enable TypeScript compilation, you need to add the
@babel/preset-typescript
preset to your Babel configuration (babel.config.js
or .babelrc
).// babel.config.js module.exports = { presets: [ "@babel/preset-env", "@babel/preset-react", "@babel/preset-typescript" ], ... };
Install Babel Presets: Ensure you have the necessary Babel presets installed as development dependencies.
npm install --save-dev @babel/preset-env @babel/preset-react @babel/preset-typescript