es7s / holms

CLI UTF-8 to Unicode code points decomposer
https://pypi.org/project/holms
1 stars 0 forks source link
ascii cli command-line console es7s python python3 terminal unicode

Downloads PyPI Coverage Status Code style: black


CLI UTF-8 decomposer for text analysis capable of displaying Unicode code point names and categories, along with ASCII control characters, UTF-16 surrogate pair pieces, invalid UTF-8 sequences parts as separate bytes, etc.

Motivation

A necessity for a tool that can quickly identify otherwise indistinguishable Unicode code points.

Installation

With pipx (recommended)

pipx install holms

From git repository

curl -sS https://github.com/es7s/holms/blob/master/install.sh | sh

Basic usage

Usage: holms run [OPTIONS] [INPUT]

  Read data from INPUT file, find all valid UTF-8 byte sequences, decode them and display as
  separate Unicode code points. Use '-' as INPUT to read from stdin instead.
example001 example004 example002 example003
Plain text output > holms run -u - <<<'1₂³⅘↉⏨' 0 U+ 31 ▕ 1 ▏ Nd DIGIT ONE 1 U+2082 ▕ ₂ ▏ No SUBSCRIPT TWO 4 U+ B3 ▕ ³ ▏ No SUPERSCRIPT THREE 6 U+2158 ▕ ⅘ ▏ No VULGAR FRACTION FOUR FIFTHS 9 U+2189 ▕ ↉ ▏ No VULGAR FRACTION ZERO THIRDS c U+23E8 ▕ ⏨ ▏ So DECIMAL EXPONENT SYMBOL > holms run -u - <<<'🌯👄🤡🎈🐳🐍' 00 U1F32F ▕🌯 ▏ So BURRITO 04 U1F444 ▕👄 ▏ So MOUTH 08 U1F921 ▕🤡 ▏ So CLOWN FACE 0c U1F388 ▕🎈 ▏ So BALLOON 10 U1F433 ▕🐳 ▏ So SPOUTING WHALE 14 U1F40D ▕🐍 ▏ So SNAKE > holms run -u - <<<'aаͣāãâȧäåₐᵃa' 00 U+ 61 ▕ a ▏ Ll LATIN SMALL LETTER A 01 U+ 430 ▕ а ▏ Ll CYRILLIC SMALL LETTER A 03 U+ 363 ▕ ͣ ▏ Mn COMBINING LATIN SMALL LETTER A 05 U+ 101 ▕ ā ▏ Ll LATIN SMALL LETTER A WITH MACRON 07 U+ E3 ▕ ã ▏ Ll LATIN SMALL LETTER A WITH TILDE 09 U+ E2 ▕ â ▏ Ll LATIN SMALL LETTER A WITH CIRCUMFLEX 0b U+ 227 ▕ ȧ ▏ Ll LATIN SMALL LETTER A WITH DOT ABOVE 0d U+ E4 ▕ ä ▏ Ll LATIN SMALL LETTER A WITH DIAERESIS 0f U+ E5 ▕ å ▏ Ll LATIN SMALL LETTER A WITH RING ABOVE 11 U+2090 ▕ ₐ ▏ Lm LATIN SUBSCRIPT SMALL LETTER A 14 U+1D43 ▕ ᵃ ▏ Lm MODIFIER LETTER SMALL A 17 U+FF41 ▕a ▏ Ll FULLWIDTH LATIN SMALL LETTER A > holms run -u - <<<'%‰∞8᪲?¿‽⚠⚠️' 00 U+ 25 ▕ % ▏ Po PERCENT SIGN 01 U+2030 ▕ ‰ ▏ Po PER MILLE SIGN 04 U+221E ▕ ∞ ▏ Sm INFINITY 07 U+ 38 ▕ 8 ▏ Nd DIGIT EIGHT 08 U+1AB2 ▕ ᪲ ▏ Mn COMBINING INFINITY 0b U+ 3F ▕ ? ▏ Po QUESTION MARK 0c U+ BF ▕ ¿ ▏ Po INVERTED QUESTION MARK 0e U+203D ▕ ‽ ▏ Po INTERROBANG 11 U+26A0 ▕ ⚠ ▏ So WARNING SIGN 14 U+26A0 ▕ ⚠ ▏ So WARNING SIGN 17 U+FE0F ▕ ️ ▏ Mn VARIATION SELECTOR-16

Buffering

The application works in two modes: buffered (the default if INPUT is a file) and unbuffered (default when reading from stdin). Options -b/-u explicitly override output mode regardless of the default setting.

In buffered mode the result begins to appear only after EOF is encountered (i.e., the WHOLE file has been read to the buffer). This is suitable for short and predictable inputs and produces the most compact output with fixed column sizes.

The unbuffered mode comes in handy when input is an endless piped stream: the results will be displayed in real time, as soon as the type of each byte sequence is determined, but the output column widths are not fixed and can vary as the process goes further.

Despite the name, the app actually uses tiny (4 bytes) input buffer, but it's the only way to handle UTF-8 stream and distinguish valid sequences from broken ones; in truly unbuffered mode the output would consist of ASCII-7 characters (0x00-0x7F) and unrecognized binary data (0x80-0xFF) only, which is not something the application was made for.

Configuration / Advanced usage

Options:
  -b, --buffered / -u, --unbuffered
                        Explicitly set to wait for EOF before processing the
                        output (buffered), or to stream the results in
                        parallel with reading, as soon as possible
                        (unbuffered). See BUFFERING section above for the
                        details.
  -m, --merge           Replace all sequences of repeating characters with one
                        of each, together with initial length of the sequence.
  -g, --group           Group the input by code points (=count unique), sort
                        descending and display counts instead of normal
                        output. Implies '--merge' and forces buffered ('-b')
                        mode. Specifying the option twice ('-gg') results in
                        grouping by code point category instead, while doing
                        it thrice ('-ggg') makes the app group the input by
                        super categories.
  -f, --format          Comma-separated list of columns to show (order is
                        preserved). Run 'holms format' to see the details.
  -n, --names           Display names instead of abbreviations. Affects `cat`
                        and `block` columns, but only if column in question is
                        already present on the screen. Note that these columns
                        can still display only the beginning of the attribute,
                        unless '-r' is provided.
  -a, --all             Display ALL columns.
  -r, --rigid           By default some columns can be compressed beyond the
                        nominal width, if all current values fit and there is
                        still space left. This option disables column
                        shrinking (but they still will be expanded when
                        needed).
  --decimal             Use decimal byte offsets instead of hexadecimal.
  --alt                 Use alternative notation for control characters: caret
                        notation for ASCII C0, octal notation for ASCII C1.
  --oneline             Discard all newline characters (0x0a LINE FEED) from
                        the input.
  --no-table            Do not format results as a table, just apply the
                        colors to characters (equivalent to '-f char', implies
                        '-b'). Compatible with '-merge', '--format' and even '
                        --group'.
  --no-override         Do not replace control/whitespace code point markers
                        with distinguishable characters ('▯' to '↵', '␣' etc).
                        Run 'holms legend' to see the details.
  -?, --help            Show this message and exit.

Examples

Output column selection

Option -f/--filter can be used to specify what columns to display. As an alternative, there is an -a/--all option that enables displaying of all currently available columns.

Column availability depending on operating mode
example010

Also -m/--merge option is demonstrated, which tells the app to collapse repetitive characters into one line of the output while counting them:

example005
Plain text output > holms run -m phpstan.txt 000 U+2B ▕ + ▏ Sm PLUS SIGN 001+ U+2D ▕ - ▏ Pd 27× HYPHEN-MINUS 01c U+2B ▕ + ▏ Sm PLUS SIGN 01d U+20 ▕ ␣ ▏ Zs SPACE 01e U+2B ▕ + ▏ Sm PLUS SIGN 01f+ U+2D ▕ - ▏ Pd 27× HYPHEN-MINUS 03a U+2B ▕ + ▏ Sm PLUS SIGN 03b U+ A ▕ ↵ ▏ Cc ASCII C0 [LF] LINE FEED 03c U+7C ▕ | ▏ Sm VERTICAL LINE 03d+ U+20 ▕ ␣ ▏ Zs 27× SPACE ...

Reading from pipeline

There is an official Unicode Consortium data file included in the repository for test purposes, named confusables.txt. In the next example we extract line #3620 using sed, delete all TAB (0x08) characters and feed the result to the application. The result demonstrates various Unicode dot/bullet code points:

example006
Plain text output > sed confusables.txt -Ee 'sg' -e '3620!d' |   holms run - 00 U+ B7 ▕ · ▏ Po MIDDLE DOT 02 U+1427 ▕ ᐧ ▏ Lo CANADIAN SYLLABICS FINAL MIDDLE DOT 05 U+ 387 ▕ · ▏ Po GREEK ANO TELEIA 07 U+2022 ▕ • ▏ Po BULLET 0a U+2027 ▕ ‧ ▏ Po HYPHENATION POINT 0d U+2219 ▕ ∙ ▏ Sm BULLET OPERATOR 10 U+22C5 ▕ ⋅ ▏ Sm DOT OPERATOR 13 U+30FB ▕・ ▏ Po KATAKANA MIDDLE DOT 16 U10101 ▕ 𐄁 ▏ Po AEGEAN WORD SEPARATOR DOT 1a U+FF65 ▕ ・ ▏ Po HALFWIDTH KATAKANA MIDDLE DOT 1d U+ A ▕ ↵ ▏ Cc ASCII C0 [LF] LINE FEED

Code points / categories statistics

-g/--group option can be used to count unique code points, and to compute the occurrence rate of each one:

example008
Plain text output > holms run -g ./tests/data/confusables.txt U+ 20 ▕ ␣ ▏ Zs 12.5% ███ 62732× SPACE U+ 9 ▕ ⇥ ▏ Cc 7.3% █▊ 36745× ASCII C0 [HT] HORIZONTAL TABULATION U+ 41 ▕ A ▏ Lu 6.1% █▍ 30555× LATIN CAPITAL LETTER A U+ 49 ▕ I ▏ Lu 5.2% █▏ 26063× LATIN CAPITAL LETTER I U+ 45 ▕ E ▏ Lu 5.0% █▏ 24992× LATIN CAPITAL LETTER E U+ 54 ▕ T ▏ Lu 3.7% ▉ 18776× LATIN CAPITAL LETTER T U+ 4C ▕ L ▏ Lu 3.7% ▉ 18763× LATIN CAPITAL LETTER L U+200E ▕ ▯ ▏ Cf 3.7% ▉ 18494× LEFT-TO-RIGHT MARK U+ A ▕ ↵ ▏ Cc 2.9% ▋ 14609× ASCII C0 [LF] LINE FEED U+ 43 ▕ C ▏ Lu 2.9% ▋ 14450× LATIN CAPITAL LETTER C ...

When used twice (-gg) or thrice (-ggg), the application groups the input by code point category or code point super category, respectively, which can be used e.g. for frequency domain analysis:

example011 example012
Plain text output > holms run -gg ./tests/data/confusables.txt 53.1% ██████████ 266233× Uppercase_Letter 12.5% ██▎ 62748× Space_Separator 10.2% █▉ 51356× Control 8.5% █▌ 42511× Decimal_Number 3.7% ▋ 18497× Format 3.0% ▌ 14832× Other_Letter 2.0% ▎ 9778× Math_Symbol 1.8% ▎ 9261× Close_Punctuation 1.8% ▎ 9259× Open_Punctuation 1.5% ▎ 7525× Other_Punctuation ... > holms run -ggg ./tests/data/confusables.txt 56.7% ██████████ 284074× Letter 13.9% ██▍ 69853× Other(C) 12.5% ██▏ 62750× Separator(Z) 8.5% █▌ 42796× Number 5.9% █ 29571× Punctuation 2.2% ▍ 11072× Symbol 0.2% ▏ 965× Mark

In-place type highlighting

When --format is specified exactly as a single char column: --format=char, the application omits all the columns and prints the original file contents, while highligting each character with a color that indicates its' Unicode category.

Note that ASCII control codes, as well as Unicode ones, are kept untouched and invisible.

example007
Plain text output > sed chars.txt -nEe 1,12p |   holms run --format=char - ! " # $ % & ' ( ) * + , - . / 0 1 2 3 4 5 6 7 8 9 : ; < = > ? @ A B C D E F G H I J K L M N O P Q R S T U V W X Y Z [ \ ] ^ _ ` a b c d e f g h i j k l m n o p q r s t u v w x y z { | } ~ ¡ ¢ £ ¤ ¥ ¦ § ¨ © ª « ¬ ­ ® ¯ ° ± ² ³ ´ µ ¶ · ¸ ¹ º » ¼ ½ ¾ ¿ À Á Â Ã Ä Å Æ Ç È É Ê Ë Ì Í Î Ï Ð Ñ Ò Ó Ô Õ Ö × Ø Ù Ú Û Ü Ý Þ ß à á â ã ä å æ ç è é ê ë ì í î ï ð ñ ò ó ô õ ö ÷ ø ù ú û ü ý þ ÿ

ASCII latin letters (A-Za-z) are colored in 50% gray color instead of regular white on purpose — this can be extremely helpful when the task is to find non-ASCII character(s) in an massive text of plain ASCII ones, or vice versa.

Below is a real example of broken characters which are the result of two operations being applied in the wrong order: UTF-8 decoding and URL %-based unescaping. This error is different from incorrect codepage selection errors, which mess up the whole text or a part of it; all byte sequences are valid UTF-8 encoded code points, but the result differs from the origin and is completely unreadable nevertheless.

example015

ASCII C0 / C1 details

While developing the application I encountered strange (as it seemed to be at the beginning) behaviour of Python interpreter, which encoded C1 control bytes as two bytes of UTF-8, while C0 control bytes were displayed as sole bytes, like it would have been encoded in a plain ASCII. Then there was a bit of researching done.

According to ISO/IEC 6429 (ECMA-48), there are two types of ASCII control codes (to be precise, much more, but for our purposes it's mostly irrelevant) — C0 and C1. The first one includes ASCII code points 0x00-0x1F and 0x7F (some authors also include a regular space character 0x20 in this list), and the characteristic property of this type is that all C0 code points are encoded in UTF-8 exactly the same as they do in 7-bit US-ASCII (ISO/IEC 646). This helps to disambiguate exactly what type of encoding is used even for broken byte sequences, considering the task is to tell if a byte represents sole code point or is actually a part of multibyte UTF-8 sequence.

However, C1 control codes are represented by 0x80-0x9F bytes, which also are valid bytes for multibyte UTF-8 sequences. In order to distinguish the first type from the second UTF-8 encodes them as two-byte sequences instead (0x800xC280, etc.); also this applies not only to control codes, but to all other ISO/IEC 8859 code points starting from 0x80.

With this in mind, let's see how the application reflects these differences. First command produces several 8-bit ASCII C1 control codes, which are classified as raw binary/non-UTF-8 data, while the second command's output consists of the very same code points but being encoded in UTF-8 (thanks to Python's full transparent Unicode support, we don't even need to bother much about the encodings and such):

example013
Plain text output > printf "\x80\x90\x9f" && python3 -c 'print("\x80\x90\x9f", end="")' |   holms run --names --decimal --all - ⏨0 #0 0x 80 -- ▕ ▯ ▏ NON UTF-8 BYTE 0x80 -- Binary ⏨1 #1 0x 90 -- ▕ ▯ ▏ NON UTF-8 BYTE 0x90 -- Binary ⏨2 #2 0x 9f -- ▕ ▯ ▏ NON UTF-8 BYTE 0x9F -- Binary ⏨3 #3 0x c2 80 U+80 ▕ ▯ ▏ ASCII C1 [PC] PADDING CHARACTER Latin-1 Supplem‥ Control ⏨5 #4 0x c2 90 U+90 ▕ ▯ ▏ ASCII C1 [DCS] DEVICE CONTROL STRING Latin-1 Supplem‥ Control ⏨7 #5 0x c2 9f U+9F ▕ ▯ ▏ ASCII C1 [APC] APPLICATION PROGRAM COMMAND Latin-1 Supplem‥ Control

Legend

The image below illustrates the color scheme developed for the app specifically, to simplify distinguishing code points of one category from others.

example009

Most frequently encountering control codes also have a unique character replacements, which allows to recognize them without reading the label or memorizing code point identifiers:

example014
Unicode Blocks
blocks

Changelog

CHANGES.rst