Skip to content

Commit

Permalink
resolves #57 add js build that publishes an npm package
Browse files Browse the repository at this point in the history
  • Loading branch information
mojavelinux committed Nov 24, 2024
1 parent 7a5c7bd commit 180887d
Show file tree
Hide file tree
Showing 13 changed files with 254 additions and 11 deletions.
3 changes: 3 additions & 0 deletions js/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
/.nvmrc
/dist/
/node_modules/
5 changes: 5 additions & 0 deletions js/.npmrc
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
access=public
audit=false
fund=false
omit=optional
package-lock=false
63 changes: 63 additions & 0 deletions js/README.adoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
= Asciidoctor Reducer

An Asciidoctor.js extension that reduces an AsciiDoc document containing include directives to a single AsciiDoc document by expanding the includes reachable from the parent document.
Additionally, the extension evaluates preprocessor conditionals (unless the option to preserve them is enabled), only keeping those lines from conditions which are true.
If the document does not contain any preprocessor directives, the extension provides access to the unmodified source.

(The CLI and API are not currently available in the JavaScript version).

== Install

This package depends on the `asciidoctor` package (>= 2.2.0, < 3.0.0), but doesn't declare it as a dependency.
Therefore, you must install that package when installing this one.

$ npm i asciidoctor @asciidoctor/reducer

If you're using the extension with Antora, there's no need to install the `asciidoctor` package as Antora provides it.

== Usage

=== Extension

You can use this extension in combination with the load API provided by Asciidoctor.
If you want to register the extension globally, require the library as follows:

[,js]
----
const Asciidoctor = require('asciidoctor')()
require('@asciidoctor/reducer').register()
----

When you use the Asciidoctor load API, the document will automatically be reduced.
You can access the reduced source by calling either the `getSource()` or `getSourceLines()` on the loaded document.

[,js]
----
const doc = Asciidoctor.loadFile('main.adoc', { safe: 'safe' })
console.log(doc.getSource())
----

You can pass a registry instance to the `register` method to register the extension with a scoped registry (scoped to the load API call).

[,js]
----
const Asciidoctor = require('asciidoctor')()
const registry = Asciidoctor.Extensions.create()
require('@asciidoctor/reducer').register(registry)
const doc = Asciidoctor.loadFile('main.adoc', { extension_registry: registry, safe: 'safe' })
----

You can also require `@asciidoctor/reducer/extensions` to access the `Extensions` class.

== Copyright and License

Copyright (C) 2021-present Dan Allen and the individual contributors to this project.
Use of this software is granted under the terms of the MIT License.

== Trademarks

AsciiDoc(R) and AsciiDoc Language(TM) are trademarks of the Eclipse Foundation, Inc.
5 changes: 5 additions & 0 deletions js/lib/extensions.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
'use strict'

require('../dist')

module.exports = Opal.Asciidoctor.Reducer.Extensions
5 changes: 5 additions & 0 deletions js/lib/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
'use strict'

const Extensions = require('./extensions')

module.exports.register = (registry) => Extensions.$register(registry)
27 changes: 27 additions & 0 deletions js/npm/transpile.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
'use strict'

const { env: ENV } = require('node:process')
const fs = require('node:fs')
const ospath = require('node:path')

let opalCompilerPath = 'opal-compiler'
try {
require.resolve(opalCompilerPath)
} catch {
const npxInstallDir = ENV.PATH.split(':')[0]
if (npxInstallDir?.endsWith('/node_modules/.bin') && npxInstallDir.startsWith(ENV.npm_config_cache + '/')) {
opalCompilerPath = require.resolve('opal-compiler', { paths: [ospath.dirname(npxInstallDir)] })
}
}

const transpiled = require(opalCompilerPath).Builder
.create()
.build('../lib/asciidoctor/reducer/conditional_directive_tracker.rb')
.build('../lib/asciidoctor/reducer/include_directive_tracker.rb')
.build('../lib/asciidoctor/reducer/header_attribute_tracker.rb')
.build('../lib/asciidoctor/reducer/preprocessor.rb')
.build('../lib/asciidoctor/reducer/tree_processor.rb')
.build('../lib/asciidoctor/reducer/extensions.rb')
.toString()
fs.mkdirSync('dist', { recursive: true })
fs.writeFileSync('dist/index.js', transpiled)
49 changes: 49 additions & 0 deletions js/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
{
"name": "@asciidoctor/reducer",
"version": "1.1.0",
"description": "An Asciidoctor.js extension to reduce an AsciiDoc document containing includes and conditionals to a single AsciiDoc document.",
"license": "MIT",
"author": "Dan Allen",
"contributors": [
"Dan Allen <dan@opendevise.com>"
],
"repository": "github:asciidoctor/asciidoctor-reducer",
"bugs": {
"url": "https://github.com/asciidoctor/asciidoctor-reducer/issues"
},
"scripts": {
"build": "npx -y --package opal-compiler@1.0.13 node npm/transpile.js",
"preci": "npm i",
"ci": "npm run build",
"postci": "npm test",
"clean": "npx rimraf dist node_modules",
"postpublish": "npx -y downdoc --postpublish",
"prepublishOnly": "npx -y downdoc --prepublish",
"test": "node --test test/*-test.js"
},
"main": "lib/index.js",
"exports": {
".": "./lib/index.js",
"./extensions": "./lib/extensions.js",
"./dist/*": "./dist/*",
"./package.json": "./package.json"
},
"devDependencies": {
"@asciidoctor/core": "~2"
},
"files": [
"bin",
"dist",
"lib"
],
"engines": {
"node": ">=16.0.0"
},
"keywords": [
"asciidoc",
"asciidoctor",
"extension",
"include",
"preprocessor"
]
}
1 change: 1 addition & 0 deletions js/test/fixtures/smoke-include.adoc
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Text in include.
7 changes: 7 additions & 0 deletions js/test/fixtures/smoke.adoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
= Smoke Test

Text in main document.

include::smoke-include.adoc[]

Text in main document.
42 changes: 42 additions & 0 deletions js/test/reducer-smoke-test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
'use strict'

const assert = require('node:assert/strict')
const { describe, before, after, it } = require('node:test')
const Asciidoctor = require('@asciidoctor/core')()
const ospath = require('node:path')

const FIXTURES_DIR = ospath.join(__dirname, 'fixtures')

describe('reducer smoke test', () => {
it('should reduce document with includes', async () => {
const registry = Asciidoctor.Extensions.create()
require('@asciidoctor/reducer').register(registry)
const input = ospath.join(FIXTURES_DIR, 'smoke.adoc')
const expected = [
'= Smoke Test',
'',
'Text in main document.',
'',
'Text in include.',
'',
'Text in main document.',
]
const actual = Asciidoctor.loadFile(input, { extension_registry: registry, safe: 'safe' }).getSourceLines()
assert.deepEqual(actual, expected)
})

it('should provide access to header attributes defined in source', async () => {
const registry = Asciidoctor.Extensions.create()
require('@asciidoctor/reducer').register(registry)
const input = [
'= Smoke Test',
':icons: font',
':toc:',
'',
'body text',
]
const expected = { icons: 'font', toc: '' }
const actual = Asciidoctor.load(input, { extension_registry: registry, safe: 'safe' }).source_header_attributes
assert.deepEqual(actual.$$smap, expected)
})
})
10 changes: 6 additions & 4 deletions lib/asciidoctor/reducer/extensions.rb
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
# frozen_string_literal: true

require 'asciidoctor' unless defined? Asciidoctor.load
require_relative 'header_attribute_tracker'
require_relative 'preprocessor'
require_relative 'tree_processor'
unless RUBY_ENGINE == 'opal'
require 'asciidoctor' unless defined? Asciidoctor.load
require_relative 'header_attribute_tracker'
require_relative 'preprocessor'
require_relative 'tree_processor'
end

module Asciidoctor::Reducer
module Extensions
Expand Down
6 changes: 4 additions & 2 deletions lib/asciidoctor/reducer/preprocessor.rb
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
# frozen_string_literal: true

require_relative 'include_directive_tracker'
require_relative 'conditional_directive_tracker'
unless RUBY_ENGINE == 'opal'
require_relative 'include_directive_tracker'
require_relative 'conditional_directive_tracker'
end

module Asciidoctor::Reducer
class Preprocessor < ::Asciidoctor::Extensions::Preprocessor
Expand Down
42 changes: 37 additions & 5 deletions release.sh
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
#!/bin/bash

# required packages (for ubuntu:kinetic): curl git jq ruby
# required packages (for ubuntu:kinetic): curl git jq nodejs npm ruby

if [ -z "$RELEASE_RUBYGEMS_API_KEY" ]; then
echo No API key specified for publishing to rubygems.org. Stopping release.
exit 1
fi
if [ -z "$RELEASE_NPM_TOKEN" ]; then
echo No npm token specified for publishing to npmjs.com. Stopping release.
exit 1
fi
export RELEASE_BRANCH=${GITHUB_REF_NAME:-main}
if [ ! -v RELEASE_USER ]; then
export RELEASE_USER=$GITHUB_ACTOR
Expand All @@ -29,28 +33,56 @@ mkdir -p $HOME/.gem
echo -e "---\n:rubygems_api_key: $RELEASE_RUBYGEMS_API_KEY" > $HOME/.gem/credentials
chmod 600 $HOME/.gem/credentials

# configure npm client for publishing
if [ "$RELEASE_VERSION" != "${RELEASE_VERSION/-/}" ]; then
RELEASE_NPM_TAG=testing
else
RELEASE_NPM_TAG=latest
fi
if case $RELEASE_VERSION in 1.0.0-*) ;; *) false;; esac; then
RELEASE_NPM_TAG=latest
fi
echo -e "//registry.npmjs.org/:_authToken=$RELEASE_NPM_TOKEN" > $HOME/.npmrc

# release!
(
set -e
ruby tasks/version.rb
(
cd js
npm version --no-git-tag-version $RELEASE_VERSION
)
git commit -a -m "release $RELEASE_VERSION [no ci]"
HEAD_COMMIT=$(git rev-parse HEAD)
(
cd js
npm run ci
sed -i '/^\/dist\/$/d' .gitignore
git add dist
)
git commit -a -m 'add dist files for npm package'
git tag -m "version $RELEASE_VERSION" v$RELEASE_VERSION
mkdir -p pkg
gem build $GEMSPEC -o pkg/$RELEASE_GEM_NAME-$RELEASE_GEM_VERSION.gem
git push origin $(git describe --tags --exact-match)
gem push pkg/$RELEASE_GEM_NAME-$RELEASE_GEM_VERSION.gem
(
cd js
npm publish --tag $RELEASE_NPM_TAG
)
git reset --hard $HEAD_COMMIT
git push origin $RELEASE_BRANCH
#sed -i 3d README.adoc
#sed -i "$(grep -m 1 -n '^== ' CHANGELOG.adoc | cut -f1 -d:)i == Unreleased\n\n_No changes since previous release._\n" CHANGELOG.adoc
#git commit -a -m 'begin development on next version [no ci]'
#git push origin $RELEASE_BRANCH
)

exit_code=$?

# nuke npm settings
rm -f $HOME/.npmrc

# nuke gem credentials
rm -rf $HOME/.gem

# check for any uncommitted files
git status -s -b

exit $exit_code

0 comments on commit 180887d

Please sign in to comment.