Internationalization & localization analysis system for C++ code, gettext catalogs, and Windows® RC files.
Quneiform
is a command-line utility and graphical user interface that scans source and resource files to check for
various i18n and l10n issues.
Additionally, the GUI version provides pseudo-translation support for gettext catalogs (*.po files).
Refer to the manual for full documentation.
Quneiform
supports static analysis for the following:
- C code
- C++ code ('98 and modern C++)
- Windows resource files (*.rc)
- macOS application Info.plist files
Static analysis and pseudo-translation are available for:
- GNU gettext translation files (*.po)
Additionally, it offers specialized support for the following frameworks:
- wxWidgets
- Qt
- KDE
- GTK+
- Win32
- MFC
¹ Strings are considered translatable if inside of gettext,
wxWidgets, Qt, or
KDE (ki18n) i18n functions.
This includes functions and macros such as gettext()
, _()
, tr()
, translate()
,
QT_TR_NOOP()
, wxTRANSLATE()
, i18n()
, etc.
² Variables are determined to be ID variables if they are integral types with the whole word "ID" in their name.
Refer here for example usage.
The command line and GUI versions provide the following checks:
- Strings exposed for translation¹ that probably should not be. This includes (but not limited to):
- Filenames
- Strings only containing
printf()
commands - Numbers
- Regular expressions
- Strings inside of debug functions
- Formulas
- Code (used for code generators)
- Strings that contain URLs, email addresses, or phone numbers
- Strings not available for translation that possibly should be.
- Strings that contain extended ASCII characters that are not encoded.
("Danke schön" instead of "Danke sch\U000000F6n".)
Encoding extended ASCII characters is recommended for best portability between compilers. - Strings with malformed syntax (e.g., malformed HTML tags).
- Use of deprecated text macros (e.g.,
wxT()
in wxWidgets,_T()
in Win32). - Use of deprecated string functions (e.g.,
_tcsncpy()
in Win32). - Files that contain extended ASCII characters, but are not UTF-8 encoded.
(It is recommended that files be UTF-8 encoded for portability between compilers.) - UTF-8 encoded files which start with a BOM/UTF-8 signature.
It is recommended to save without the file signature for best compiler portability. printf()
-like functions being used to just format an integer to a string.
It is recommended to usestd::to_string()
to do this instead.printf()
command mismatches between source and translation strings.
(PO catalogs with C/C++ strings are currently supported.)- Font issues in Windows resource files:
- Dialogs not using "MS Shell Dlg" or "MS Shell Dlg 2."
- Dialogs with non-standard font sizes.
- Info.plist files not including any
CFBundleLocalizations
keys.
Code formatting and other issues can also be checked for, such as:
- Trailing spaces at the end of a line.
- Tabs (instead of spaces).
- Lines longer than 120 characters.
- Spaces missing between "//" and their comments.
- ID variable² assignment issues:
- The same value being assigned to different ID variables in the same source file (e.g., "wxID_HIGHEST + 1" being assigned to two menu ID constants).
- Hard-coded numbers being assigned to ID variables.
- Out-of-range values being assigned to MFC IDs.
(available in the GUI version)
Pseudo-translation includes features such as:
- Multiple methods for character replacement (e.g., replacing characters with accented variations or upper casing them).
- Increasing the width of the translations. This is useful for ensuring that strings are not being truncated at runtime.
- Wrapping the translations in braces. This is useful for ensuring that strings are not being pieced together at runtime.
- Appending a unique ID to each translation. This is useful for finding where a translation is being loaded from.
When pseudo-translating, a copy of all gettext *.po and *.pot files will be created and have their translations filled with mutations of the respective source strings. These files (which will have a "pseudo_" prefix) can then be compiled into *.mo files and loaded by your application for integration testing.
Quneiform
accepts the following arguments:
[input]: The folder to analyze.
--enable: Which checks to perform. Can be any combination of:
allI18N: Perform all internationalization checks (the default).
allL10N: Perform all localization checks (the default).
allCodeFormatting: Check all code formatting issues.
These are not enabled by default.
suspectL10NString: Check for translatable strings that shouldn't be
(e.g., numbers, keywords, printf() commands).
suspectL10NUsage: Check for translatable strings being used in internal contexts
(e.g., debugging functions).
suspectI18NUsage: Check for suspect usage of i18n functions.
urlInL10NString: Check for translatable strings that contain URLs, email addresses, or phone numbers.
It is recommended to dynamically format these into the string so that
translators don't have to manage them.
concatenatedStrings: Check for strings that may be concatenated at runtime.
L10NStringNeedsContext: Check for possibly ambiguous strings that lack a translator comment.
notL10NAvailable: Check for strings not exposed for translation.
deprecatedMacro: Check for deprecated text macros (e.g., wxT()).
nonUTF8File: Check that files containing extended ASCII characters are UTF-8 encoded.
UTF8FileWithBOM: Check for UTF-8 encoded files which start with a BOM/UTF-8 signature.
It is recommended to save without the file signature for
best compiler portability.
unencodedExtASCII: Check for strings containing extended ASCII characters that are not encoded.
printfSingleNumber: Check for printf()-like functions being used to just format a number.
dupValAssignedToIds: Check for the same value being assigned to different ID variables.
numberAssignedToId: Check for ID variables being assigned a hard-coded number.
It may be preferred to assign framework-defined constants to IDs.
malformedString: Check for malformed syntax in strings (e.g., malformed HTML tags).
fontIssue: Check for font issues (e.g., Windows *.RC dialogs not using MS Shell Dlg
or using unusual font sizes).
trailingSpaces: Check for trailing spaces at the end of each line.
This is a code formatting check and is not enabled by default.
tabs: Check for tabs.
(Spaces are recommended as tabs may appear differently between editors.)
This is a code formatting check and is not enabled by default.
wideLine: Check for overly long lines.
This is a code formatting check and is not enabled by default.
commentMissingSpace: Check that there is a space at the start of a comment.
This is a code formatting check and is not enabled by default.
printfMismatch: Check for mismatching printf() commands between source and translation strings.
acceleratorMismatch: Check for mismatching keyboard accelerators between source and translation strings.
transInconsistency: Check for inconsistent trailing punctuation, spaces, or newlines.
numberInconsistency: Check for mismatching numbers between the source and target strings.
lengthInconsistency: Check for suspect lengths of translations compared to their source strings.
excessiveNonL10NContent: Check for translatable strings that contain large blocks on non-translatable content.
halfWidth: Check for halfwidth Kanas, Hanguls, and punctuation in source and target strings.
multipartString: Check for strings that appear to contain multiple parts that are
being sliced at runtime.
pluralization: Check for strings being used for both singular and plural that
should be use different variations.
articleOrPronoun: Check for strings with an article (e.g., the, a) in front of a
formatting placeholder.
Also checks for pronouns being used as individual strings.
--disable: Which checks to not perform. (Refer to options available above.)
This will override any options passed to "--enable".
--log-l10n-allowed: Whether it is acceptable to pass translatable strings to
logging functions. Setting this to false will emit warnings
when a translatable string is passed to functions such as
wxLogMessage or SDL_Log.
(Default is true.)
--punct-l10n-allowed: Whether it is acceptable for punctuation only strings to be
translatable. Setting this to true will suppress warnings about
strings such as " - " being available for localization.
(Default is false.)
--exceptions-l10n-required: Whether to verify that exception messages are available
for translation. Setting this to true will emit warnings
when untranslatable strings are passed to various exception
constructors or functions (e.g., AfxThrowOleDispatchException).
(Default is true.)
--min-l10n-wordcount: The minimum number of words that a string must have to be
considered translatable. Higher values for this will result in less
strings being classified as a notL10NAvailable warning.
(Default is 2.)
--fuzzy: Whether to review fuzzy translations.
(Default is false.)
--cpp-version: The C++ standard that should be assumed when issuing deprecated macro warnings.
(Default is 2014.)
-i,--ignore: Folders and files to ignore (can be used multiple times).
-o,--output: The output report path (tab-delimited or CSV format).
(Can either be a full path, or a file name within the current working directory.)
-q,--quiet: Only print errors and the final output.
-v,--verbose: Perform additional checks and display debug information.
-h,--help: Print usage.
The following example will analyze the folder "c:\src\wxWidgets\src" (but ignore the subfolders "expat" and "zlib"). It will only check for suspect translatable strings, and then send the output to "results.txt" in the current working directory.
quneiform C:\src\wxWidgets\src -i expat -i zlib --enable=suspectL10NString -o results.txt
This example will only check for suspectL10NUsage
and suspectL10NString
and not show
any progress messages.
quneiform C:\src\wxWidgets\samples -q --enable=suspectL10NUsage,suspectL10NString
This example will ignore multiple folders (and files) and output the results to "WDVResults.txt."
quneiform C:\src\Wisteria-dataviz\src --ignore=import,wxsimplejson,wxStartPage,math,easyexif,debug,utfcpp,CRCpp,base/colors.cpp,base/colors.h -o WDVresults.txt
Refer here for more examples.
Quneiform
can be configured and built with Cmake.
On Unix:
cmake . -DCMAKE_BUILD_TYPE=Release
cmake --build . --target all -j $(nproc) --config Release
On Windows, "CMakeLists.txt" can be opened and built directly in Visual Studio.
After building, "quneiform" will be available in the "bin" folder.
wxWidgets 3.3 or higher is required for building the graphical user interface version.
Download wxWidgets, placing it at the same folder level as this project. After building wxWidgets, Quneiform can be configured and built with Cmake.
On Unix:
cd ..
git clone /~https://github.com/wxWidgets/wxWidgets.git --recurse-submodules
cd wxWidgets
cmake . -DCMAKE_INSTALL_PREFIX=./wxlib -DwxBUILD_SHARED=OFF \
-D"CMAKE_OSX_ARCHITECTURES:STRING=arm64;x86_64" \
-DCMAKE_OSX_DEPLOYMENT_TARGET=10.15 \
-DwxBUILD_OPTIMISE=ON -DwxBUILD_STRIPPED_RELEASE=ON \
-DCMAKE_BUILD_TYPE=Release
cmake --build . --target install --config Release
cd ..
cd Quneiform/gui
cmake . -DCMAKE_BUILD_TYPE=Release
cmake --build . --target all --config Release
On Windows with Visual Studio, build wxWidgets with the defaults, except wxBUILD_SHARED
should be set to "OFF"
(and MAKE_BUILD_TYPE
set to "Release" for release builds).
Open "gui/CMakeLists.txt" in Visual Studio, setting the CMake setting's configuration type to "Release" for a release build.
After building, "quneiform" will be available in the "bin" folder.
You can also create a Quneiform
GitHub action to make it part of your CI.
For example, create a new workflow called "quneiform.yml" and enter the following:
name: Quneiform
on: [push]
jobs:
build:
name: Quneiform
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: install Quneiform
run: |
git clone /~https://github.com/Blake-Madden/Quneiform.git --recurse-submodules
cd Quneiform
cmake ./
make -j4
cd ..
- name: analyze
run: |
# Ignore Quneiform's own folder.
# You can ignore other folders by adding a comma and the folder name
# after "--ignore=Quneiform".
./Quneiform/bin/quneiform ./ --ignore=Quneiform -q -o i18nresults.txt
- name: review results
run: |
REPORTFILE=./i18nresults.txt
WARNINGSFILE=./warnings.txt
if test -f "$REPORTFILE"; then
cat "$REPORTFILE" > "$WARNINGSFILE"
# are there any warnings?
if grep -qP '\[[a-zA-Z0-9]+\]' "$WARNINGSFILE"; then
# print the remaining warnings
echo Warnings detected:
echo ==================
cat "$WARNINGSFILE" | grep -P '\[[a-zA-Z0-9]+\]'
# fail the job
exit 1
else
echo No issues detected
fi
else
echo "$REPORTFILE" not found
fi