meTypeset is a tool to covert from Microsoft Word .docx format to NLM/JATS-XML for scholarly/scientific article typesetting.
meTypeset is a fork of the OxGarage stack and uses TEI as an intemediary format to facilitate interchange. Components are licensed under the GPL v2 and the Creative Commons Attribution-Sharealike 3.0 clauses, according to the various original sources of the fork.
The transforms within this software can be invoked in several different ways:
-
Run through a *nix compatible command line with the bin/meTypeset.py script.
-
A limited set of the transforms can be called through any renderer capable of parsing XSLT 1.0. This will not include the meTypeset classifier stages.
The modfications to the OxGarage stack contained within this project are Copyright Martin Paul Eve 2014 and released under the licenses specified in each respective file.
- Automated Microsoft Word (docx) to JATS XML
- Intelligent size processing and section grouping algorithm
- Reference list detection
- Free-text list detection
- Footnote handling
- Full table support
- Metadata handling from platform-generated input
- Built-in bibliographic database
- Zotero integration for looking up citations
- Free software
First off, ensure you have the correct stack installed for your environment. meTypeset needs a valid python 3 environment, a valid Java environment, the command line tools "unzip", "basename", "unoconv" and a shell interpreter (eg Bash). Saxon, which is required, is pre-bundled. It has been shown to work correctly on *Nix derivatives and Mac OS X. Please see the file INSTALL.md for @ppKrauss's Ubuntu installation instructions.
meTypeset: text parsing library to convert word documents to the JATS XML format
Usage:
meTypeset.py doc <input> <output_folder> [options]
meTypeset.py docx <input> <output_folder> [options]
meTypeset.py docxextracted <input> <output_folder> [options]
meTypeset.py odt <input> <output_folder> [options]
meTypeset.py other <input> <output_folder> [options]
meTypeset.py tei <input> <output_folder> [options]
meTypeset.py bibscan <input> [options]
Options:
-a, --aggression <aggression_level> Parser aggression level 0-10 [default: 10]
--chain <xslt> Specify a subsequent XSL transform to pass the NLM to
-c, --clean Produce final XML, not intermediate markup with additional metadata
-d, --debug Enable debug output
-i, --identifiers Generate unique identifiers for all supported NLM elements
--includedeleted Keep deleted text (track changes)
--interactive Enable step-by-step interactive mode
-h, --help Show this screen.
-m, --metadata <metadata_file> Metadata file
--nogit Disable git debug filesystem (only of use with --debug)
--noimageprocessing Disable unoconv image processing
--nolink Do not run reference linker
--nometa Do not merge front matter
--purenlm Die after performing NLM XSLT step
--puretei Die after performing TEI XSLT step
--prettytei Indent and format intermediary TEI
-p, --proprietary Enables proprietary math parsing. Requires omml2mml.xsl
-s, --settings <settings_file> Settings file
-v, --version Show version.
-z, --zotero Enable Zotero integration for references.
When running with the doc command, input should be a Word DOC file.
When running with the docx command, input should be a Word DOCX file.
When running with the docxextracted command, input should be a folder containing the usually compressed contents of a Word DOCX file (unzip x.docx).
When running with the odt command, input should be a OpenOffice/LibreOffice ODT file.
When running with the other command, input should be any file that unoconv can covert to a word DOCX file.
When running with the bibscan command, input should be an NLM XML file, from which bibliographic reference information will be extracted.
Note well that best results come from Word DOCX file and the "doc", "ODT" and "other" commands are provided as helper methods but rely on unoconv to correctly convert the file.
If you would like to enable bash completion for meTypeset, simply copy the bash/meTypeset.sh file into your /etc/bash_completion.d folder.
Occasionally, meTypeset will encounter a document with a greater number of idiosyncracies than it can handle. In this event, an error log will be written to "errors.txt" in the path specified by the mt:error configuration value. This document consists of a list of errors, each line beginning with a [ERROR_NUMBER] text (eg [001]).
Furthermore, the parser will attempt to append an attribute named "rend" to the tags that it felt most problematic with the error number specified within (eg <p rend="error-001">. This is designed to aid visual tools in analysing the problematic areas.
- 001 - the number of linebreaks found in the document exceeded 80. The parser will mark up elements containing more than 3 comments with rend="error-001"
- 002 - a serious problem occurred in the size classifier that would disorder the document. The parser will revert to single-level headings.
Some portions of the meTypeset stack can be run independently of the main application.
The reference linker will scan an NLM document for paranthetical references and match them up to ref elements.
If called with the "scan" option, the reference linker will attempt to match text in brackets to likely reference list elements.
If called with the "link" option, the reference linker will change the "rid" attribute on the xref element with an "id" attribute equal to "source_id" to the value of "dest_id".
referencelinker.py: a tool to link parenthetical references to ref-list elements in a JATS file
Usage:
referencelinker.py scan <input> [options]
referencelinker.py link <input> <source_id> <dest_id> [options]
referencelinker.py prune <input> [options]
Options:
-d, --debug Enable debug output
--interactive Prompt the user to assist in interactive tagging
-h, --help Show this screen.
-v, --version Show version.
-z, --zotero Enable Zotero integration for references.
The NLM processor will run the specific post-TEI stages of meTypeset. This is most useful for reference linking, table/graphic heading/caption classification and document cleanup.
nlmprocessor.py: a tool to run the NLM portions of meTypeset
Usage:
nlmprocessor.py process <input> [options]
Options:
-a, --aggression <aggression_level> Parser aggression level 0-10 [default: 10]
--chain <xslt> Specify a subsequent XSL transform to pass the NLM to
-c, --clean Produce final XML, not intermediate markup with additional metadata
-d, --debug Enable debug output
-h, --help Show this screen.
-i, --identifiers Generate unique identifiers for all supported NLM elements
--interactive Enable step-by-step interactive mode
--nogit Disable git debug filesystem (only of use with --debug)
--noimageprocessing Disable unoconv image processing
-s, --settings <settings_file> Settings file
-v, --version Show version.
-z, --zotero Enable Zotero integration for references.
I am getting errors like "UnicodeEncodeError: 'ascii' codec can't encode character u'\xa0' in position 20: ordinal not in range(128)"
Solution: Set the environment variable PYTHONIOENCODING=UTF-8
meTypeset works on a modular basis. The initial call to meTypeset creates a new instance of the meTypeset class. This, in turn, initializes the debugger and the global settings variable class. It also parses the command line options using docopt.
The main module then sequentially calls the submodules and classifiers that handle different processes in the parsing of the document. Whether or not these modules are called is defined by the --aggression switch. Aggression levels correlate to the risk involved in running each of the procedures as some are more statistically likely to fail and also to falsely classify other parts of the document.
If the command argument given is docx or docxextracted, the first call is to the DOCX to TEI parser. This module extracts the docx file (if argument was "docx") to a temporary folder, copies across all necessary transform stylesheets, extracts any media files embedded in the word document to the "media" folder and then calls Saxon to perform an initial transform to TEI format.
If the appropriate aggression level is set, the next step is to proceed to the Size Classifier. This module handles classification of sizes and headings within the document. Taking a given minimum size cutoff (16) as a basis, it classifies text above this level as a heading, so long as no more than 40 headings of this size exist in a document. It then proceeds to organize these headings into different nested sub-levels using a TEI-Manipulator object to do the heavy lifting. The procedure for all this is as follows:
- Classify all bold headings as root-level titles
- Process well-formed Word headings and correlate them to other sizes in the document (i.e. Word's "heading 2" correlates to the second largest size heading specified elsewhere by font size, for example)
- Organize these headings into sections, including downsizing of heading elements that try to be larger than the root node
While properly formatted Word lists are handled in the XML transforms, we often encounter documents where the user has employed a variety of homegrown stylings, such as successive paragraphs beginning with (1) etc. This module has three steps:
- Classify lists beginning with -
- Classify lists beginning with (1)
- Classify reference lists and footnotes parsed as [1]
The final step is the most difficult. If we find a list at the end of the document in the [1] format that is /not referenced elsewhere/, then we treat it as a bibliography. If, however, we find a list in this format, but then additional mentions of [1] and [2] earlier in the document, then we assume that this is ad-hoc footnote syntax and convert to that, instead.
The bibliographic addins classifier processes Zotero and Mendeley references, enclosing them within the format that the bibliography classifier can then subsequently handle. It then optionally drops any other addins that we don't know about to avoid courrupted text and JSON strings in the document.
The bibliography classifier converts marked-up paragraphs into proper bibliography format ready for post-processing and reference parsing.
The TEI Manipulator then changes WMF files into PNG images for maximum compatibility and removes any empty tags and sections that consist /only/ of title elements.
The TEI to NLM transform procedure is then called, which as with the DOCX to TEI portion calls Saxon on a stylesheet.
The metadata merge merges in a metadata heading with the NLM. Ideally, this is produced by a plugin in your journal/content management system.
The bibliographic database inserts fully marked-up JATS element-citation blocks for citations that it has encountered previously. These can be imported by using the "bibscan" command.
If the user has specified the --zotero option and set the mt:zotero setting to a valid local Zotero database, the bibliographic database will look up citations from Zotero instead of its internal database. meTypeset will place the original citation given by the user into a comment immediately below the reference found in Zotero. References from Zotero will only be imported if one and only one match is found for a string. meTypeset will begin with the whole reference string from the original document, then sequentially eliminate words until there are only three left. If it has not found a Zotero reference by this point, it will leave the original mixed-citation in place.
Finally, an optional additional XSL file can be specified to be transformed by passing the --chain option.
We are building a suite of regression tests to be supplemented every time we fix a bug in the parser. This ensures that future fixes don't undo our work so far. If you find a bug, please consider creating the smallest possible document that you feel can demonstrate this so that we can write a test for it before fixing.
The parser works best with good input. If you use the provided styles for headings in Word/OpenOffice/LibreOffice then meTypeset will correctly nest these up to nine levels deep. The parser will also handle tables, graphics and other elements, although captions are best detected when placed immediately before or after the element in question. Bibliographies are best detected when placed at the end of the document, either in a section entitled "References" (or similar) or formatted properly with a single line per element with a date in each. Alternatively, you can use a bibliographic manager such as Zotero or Mendeley and these will be handled. The reference linker works best with paranthetical references (Eve 2014) that correlate to a unique bibliographic entry.
The parser attempts to parse headings that are all capitals, all-bold or of a substantially different size to the body text. It may, however, have difficulty nesting these. The parser will not handle drawings or aligned formatting made in a Word document; the JATS/NLM spec simply doesn't provide for this. Don't, therefore, draw ascii art tables and expect them to parse! Likewise, including extensive front-matter in a document may cause problems and it is often best removed. meTypeset does not yet handle front-matter parsing, so this can cause problems for the size classifier/nesting mechanism.
meTypeset wraps around a fork of the TEI Consortium's OxGarage stylesheets.
The lead developer is Professor Martin Paul Eve (martin.eve@bbk.ac.uk).
Additional contributions were made, in alphabetical order) by: