Transformers

See also the plugin-utils package for utility functions.

API

Each transformer must implement the following API:

import { Source, TransformContext } from "html-validate";

/* implementation */
export function myTransform(this: TransformContext, source: Source): Iterable<Source> {
  /* ... */
  return [];
}

/* api version declaration */
myTransform.api = 1;

/* export transformer */
export default myTransform;

TransformContext

export interface TransformContext {
  hasChain(filename: string): boolean;
  chain(source: Source, filename: string): Iterable<Source> | Promise<Iterable<Source>>;
}

chain

Chain transformations. Sometimes multiple transformers must be applied. For instance, a Markdown file with JSX in a code-block.

export async function myTransform(
  this: TransformContext,
  source: Source,
): Promise<Iterable<Source>> {
  const sources = [];
  for (const transformedSource of transformImplementation(source)) {
    const next = `${source.filename}.foo`;
    for (const chained of await this.chain(transformedSource, next)) {
      sources.push(chained);
    }
  }
  return sources;
}

The above snippet will chain transformations using the current transformer matching *.foo files, if it is configured.

If no pattern matches the filename the input source is returned unmodified. Use hasChain to test if chaining is present.

hasChain

Test if an additional chainable transformer is present. Returns true only if there is a transformer configured for the given filename.

While it is always safe to call chain(..) as it will passthru sources without a chainable transform it is sometimes desirable to control whenever a Source should be yielded or not by determining if the user has configured a transformer or not.

Given a configuration such as:

{
  "transform": {
    "^.*\\.foo$": "my-transformer",
    "^.*:virtual$": "my-other-transformer"
  }
}

my-transformer can then implement the following pattern:

export function myTransform(
  this: TransformContext,
  source: Source,
): Iterable<Source> | Promise<Iterable<Source>> {
  /* create a virtual filename */
  const next = `${source.filename}:virtual`;
  if (this.hasChain(next)) {
    return this.chain(source, next);
  } else {
    return [source];
  }
}

By letting the user configure the .*:virtual pattern the user can control whenever my-transformer will yield a source for the match or not. This is useful when the transformer needs to deal with multiple languages and the user should ultimately be able to control whenever a language should be validated by HTML-validate or not.

Source

The validator engine works on a Source object. A transformer take a source object as argument and returns zero or more new sources.

export interface Source {
  data: string;
  filename: string;
  line: number;
  column: number;
  offset: number;
  originalData?: string;
  hooks?: SourceHooks;
}

The data property is the markup/source code for the source object. Since the data property might be only a small part of the original file there is also the related property originalData which is the full markup/source code of the file. line, column and offset is the starting location of the data relative to originalData. Line and column start at 1 and offset start at 0. filename is always the original filename. If the transformer wants to apply hooks for later processing they are set directly on the hooks property. Hooks should only be added in the last transformer, if chaining is used the hook might be overwritten or ignored.

Source hooks

Transformers can add hooks for additional processing by setting source.hooks:

import { AttributeData, HtmlElement } from "html-validate";

function processAttribute(attr: AttributeData): IterableIterator<AttributeData> {
  /* handle attribute */
  return [];
}

function processElement(node: HtmlElement): void {
  /* handle element */
}

source.hooks = {
  processAttribute,
  processElement,
};

processAttribute

Called before an attribute is set on HtmlElement and can be used to modify both the key and value. If the attribute is processed with scripting (e.g. databinding) the value may be replaced with DynamicValue.

processElement

Called after element is fully created but before children are parsed. Can be used to manipulate elements (e.g. add dynamic text from frameworks).