This package aims to generate documents by processing another one, allowing throughout the process to verify the input document.
npm i strawberry-processor
Here you have a Hello World! example:
- Input file README.toml
title="Toml Example"
message="Hello, World!"
Now you can create a processor to convert it to a Markdown file:
class SimpleProcessor extends DocumentObserver(DocumentProcessor) {
@process("title")
public title(title: string): string {
return `# ${title}`;
}
@process("message" )
public message(message: string): string {
return `__${message}__`;
}
}
Finally, you have to run the processor:
const provider = new JSONFileProvider("README.toml")
const outputHandler = new FileOutputHandler("README.md")
const processor = new SimpleProcessor(provider, outputHandler);
processor.start()
.then(() => console.log("Finish"))
.catch((error) => console.error(error));
And you got this:
# Toml Example
__Hello, World!__
You can see more examples inside the examples folder
For example, it is useful when you have to generate multiple documents with the same structure. If you have to keep different markdown files or other types of files in multiple repositories ,as can be web components, and you want each file to have the same structure and design, you can write a json or toml file with the content and forget about the document design. The design will be applied with this tool and a review step is also being added to verify the correctness of the document as shown below.
You may want to check the correctness of a file and it cannot be checked by itself as a markdown file. With this package, you can add rules to check every part of the document you want.
To add a requirement to a field you can add extra params to the process decorator.
@process(id: string, requirement?: FieldRequirement)
interface FieldRequirement {
mandatory?: boolean;
requirement?: (field: any) => boolean;
explanation?: (field: any) => string;
}
So, you can do things like that:
- Do a field optional, fields are mandatory by default:
@process("message", { mandatory: false } )
public message(message: string): string {
return `__${message}__`;
}
- Add a custom check
@process("message", { requirement: (message: string) => message.lenght > 10 } )
public message(message: string): string {
return `__${message}__`;
}
- Reuse a check
@process("author", minSize(15) )
public author(author: string): string {
return `__${author}__`;
}
@process("message", minSize(100) )
public message(message: string): string {
return `__${message}__`;
}
function minSize(size: number): FieldRequirement {
return {
explanation: (text: string) => `should have at least ${size} characters and it only has ${text.length}`,
requirement : (text: string) => text.length > size
}
}
If any of these requirements fail the processor will throw an error with an explanation.
For example, for this input:
{
"title" : "Less than 10",
"message": "Less than 100"
}
You got this result:
Field [title] should have at least 15 characters and it only has 12
Field [message] should have at least 100 characters and it only has 13
There are some errors in the input document
As you can see, it can be useful to add to your continuous integration so you can easily verify the correctness and deny any PR.
To process a children field you have to use the period syntax, as shown below:
- We will process the author name so we need to specify this id: author.name:
title = "Make room, Make room"
description = "It is a 1966 science fiction novel exploring the consequences of unchecked population growth on society."
[author]
name = "Harry"
lastName = "Harrison"
@process("author.name")
public message(authorName: string): string {
return `__${authorName}__`;
}
You can receive an object instead of a string, for example:
@process("author")
public message(author: {name: string, lastName: string}): string {
return `__${name} ${lastName}__`;
}
If you need to get a field when you are processing another you can use the field method, you can also use the period syntax with the field method.
@process("author")
public message(author: {name: string, lastName: string}): string {
const title = this.field("title");
return `__${name} ${lastName}__ is the author of ${title}`;
}
The field method is parametrized so you can specify a return type:
@process("author")
public message(author: {name: string, lastName: string}): string {
const title = this.field<string>("title");
return `__${name} ${lastName}__ is the author of ${title}`;
}
To register an error you can use the error method and add a new one.
@process("title")
public title(title: string): string {
// ...
this.addError(new FieldError("Custom error!"));
// ...
}
Processors are async/await friendly so you can specify a process method like an async method and return a promise:
@process("title")
public async title(title: string): Promise<string> {
const processedTitle = await convertTitle(title);
return `# ${processedTitle}`
}
Due to processors are async/await friendly you can use others processor when you are processing your document, as shown in the example below:
{
"title" : "JSON Example",
"anotherFile": "example.md"
}
@process("anotherFile")
public async title(anotherFile: string): Promise<string> {
const provider = new JSONFileProvider(anotherFile)
const outputHandler = new ContentOutputHandler()
await AnotherProcessor(provider, outputHandler).start()
return outputHandler.content;
}
This package has the following provider that you can use to read your input documents if you want your custom provider you can implement the Provider interface.
- JSONProvider, JSONFileProvider and TomlFileProvider .
And it has the following output handler, you can also write your output handler by implementing the OutputHandler interface.
- DefaultOutputHandler prints the result on the console.
- FileOutputHandler saves the result in a file.
- ContentOutputHandler saves the result in its content field.