Skip to content

Commit

Permalink
Added: language support (#270)
Browse files Browse the repository at this point in the history
An environment variable LANG needs to contain a GNU posix locale
for it to pick up the right language file.

- Added a functional test for it too.

- Now we look into other platforms also and show the page
along with a warning message that this page is from
a different platform.

This follows the spec more closely.
  • Loading branch information
agnivade authored Nov 20, 2019
1 parent 48d6c65 commit f2b1085
Show file tree
Hide file tree
Showing 9 changed files with 232 additions and 70 deletions.
1 change: 0 additions & 1 deletion .eslintrc.json
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@
"arrow-parens": [2, "always"],
"arrow-body-style": [2, "always"],
"array-callback-return": 2,
"complexity": [2, 10],
"no-magic-numbers": [2, {
"ignore": [-1, 0, 1, 2],
"ignoreArrayIndexes": true,
Expand Down
24 changes: 16 additions & 8 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ To see tldr pages:
- `tldr <command> --os=<platform>` show command page for the given platform (`linux`, `osx`, `sunos`)
- `tldr --search "<query>"` search all pages for the query
- `tldr --linux <command>` show command page for Linux
- `tldr --osx <command>` show command page for OSX
- `tldr --osx <command>` show command page for OSX
- `tldr --sunos <command>` show command page for SunOS
- `tldr --list` show all pages for current platform
- `tldr --list-all` show all available pages
Expand All @@ -59,6 +59,14 @@ As a contributor, you might also need the following commands:

- `tldr --render <path>` render a local page for testing purposes

Tldr pages defaults to showing pages in the current language of the operating system, or English if that's not available. To view tldr pages for a different language, set an environment variable `LANG` containing a valid [POSIX locale](https://www.gnu.org/software/gettext/manual/html_node/Locale-Names.html#Locale-Names) (such as `zh`, `pt_BR`, or `fr`) and then run the above commands as usual. In most `*nix` systems, this variable will already be set.

It is suggested that the `LANG` environment variable be set system-wide if this isn't already the case. Users without `sudo` access can set it locally in their `~/.profile`.

- `LANG=zh tldr <command>`

For the list of available translations, please refer to the main [tldr](https://github.com/tldr-pages/tldr) repo.

## Configuration

You can configure the `tldr` client by adding a `.tldrrc` file in your HOME directory. You can copy the contents of the `config.json` file from the repo to get the basic structure to start with, and modify it to suit your needs.
Expand Down Expand Up @@ -147,25 +155,25 @@ fpath = (my/completions $fpath)

#### Installation Issues

- If you are trying to install as non-root user (`npm install -g tldr`) and get something like -
- If you are trying to install as non-root user (`npm install -g tldr`) and get something like:

```
Error: EACCES: permission denied, access '/usr/local/lib/node_modules/tldr'
```

Then most probably your npm's default installation directory has improper permissions. You can resolve it by clicking [here](https://docs.npmjs.com/getting-started/fixing-npm-permissions)
- If you are trying to install as a root user (`sudo npm install -g tldr`) and get something like -

- If you are trying to install as a root user (`sudo npm install -g tldr`) and get something like:

```
as root ->
as root ->
gyp WARN EACCES attempting to reinstall using temporary dev dir "/usr/local/lib/node_modules/tldr/node_modules/webworker-threads/.node-gyp"
gyp WARN EACCES user "root" does not have permission to access the dev dir "/usr/local/lib/node_modules/tldr/node_modules/webworker-threads/.node-gyp/8.9.1"
```

You need to add the option `--unsafe-perm` to your command. This is because when npm goes to the postinstall step, it downgrades the permission levels to "nobody". Probably you should fix your installation directory permissions and install as a non-root user in the first place.

- If you see an error related to `webworker-threads` like -
- If you see an error related to `webworker-threads` like:

```
/usr/local/lib/node_modules/tldr/node_modules/natural/lib/natural/classifiers/classifier.js:32
Expand Down
8 changes: 6 additions & 2 deletions lib/cache.js
Original file line number Diff line number Diff line change
Expand Up @@ -22,12 +22,16 @@ class Cache {

getPage(page) {
let preferredPlatform = platform.getPreferredPlatformFolder(this.config);
return index.findPlatform(page, preferredPlatform)
let preferredLanguage = 'en';
if (process.env['LANG'] !== '') {
preferredLanguage = process.env['LANG'];
}
return index.findPage(page, preferredPlatform, preferredLanguage)
.then((folder) => {
if (!folder) {
return;
}
let filePath = path.join(this.cacheFolder, 'pages', folder, page + '.md');
let filePath = path.join(this.cacheFolder, folder, page + '.md');
return fs.readFile(filePath, 'utf8');
})
.catch((err) => {
Expand Down
103 changes: 90 additions & 13 deletions lib/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,28 +7,90 @@ const utils = require('./utils');

let shortIndex = null;

const pagesPath = path.join(config.get().cache, 'cache/pages');
const pagesPath = path.join(config.get().cache, 'cache');
const shortIndexFile = path.join(pagesPath, 'shortIndex.json');

function findPlatform(page, preferredPlatform) {
function findPage(page, preferredPlatform, preferredLanguage) {
// Load the index
return getShortIndex()
.then((idx) => {
// First, check whether page is in the index
if (! (page in idx)) {
return null;
}
// Get the platforms
let platforms = idx[page];
if (platforms.indexOf(preferredPlatform) >= 0) {
return preferredPlatform;
} else if (platforms.indexOf('common') >= 0) {
return 'common';
const targets = idx[page].targets;

// Remove unwanted stuff from lang code.
if (preferredLanguage.includes('.')) {
preferredLanguage = preferredLanguage.substring(0, preferredLanguage.indexOf('.'));
}
if (preferredLanguage.includes('@')) {
preferredLanguage = preferredLanguage.substring(0, preferredLanguage.indexOf('@'));
}

let ll;
if (preferredLanguage.includes('_')) {
ll = preferredLanguage.substring(0, preferredLanguage.indexOf('_'));
}
if (!hasLang(targets, preferredLanguage)) {
preferredLanguage = ll;
}

// Page resolution logic:
// 1. Look into the target platform, target lang
// 2. If not found, look into target platform, en lang.
// 3. If not found, look into common, target lang.
// 4. If not found, look into common, en lang.
// 5. If not found, look into any platform, target lang.
// 6. If not found, look into any platform, en lang.
let targetPlatform;
let targetLanguage;
if (hasPlatformLang(targets, preferredPlatform, preferredLanguage)) {
targetLanguage = preferredLanguage;
targetPlatform = preferredPlatform;
} else if (hasPlatformLang(targets, preferredPlatform, 'en')) {
targetLanguage = 'en';
targetPlatform = preferredPlatform;
} else if (hasPlatformLang(targets, 'common', preferredLanguage)) {
targetLanguage = preferredLanguage;
targetPlatform = 'common';
} else if (hasPlatformLang(targets, 'common', 'en')) {
targetLanguage = 'en';
targetPlatform = 'common';
} else if (targets.length > 0 && hasLang(targets, preferredLanguage)) {
targetLanguage = preferredLanguage;
targetPlatform = targets[0].os;
console.log(`Command ${page} does not exist for the host platform. Displaying the page from ${targets[0].os} platform`);
} else if (targets.length > 0 && hasLang(targets, 'en')) {
targetLanguage = 'en';
targetPlatform = targets[0].os;
console.log(`Command ${page} does not exist for the host platform. Displaying the page from ${targets[0].os} platform`);
}

if (!targetLanguage && !targetPlatform) {
return null;
}
return null;

let targetPath = 'pages';
if (targetLanguage !== 'en') {
targetPath += '.' + targetLanguage;
}
return path.join(targetPath, targetPlatform);
});
}

function hasPlatformLang(targets, preferredPlatform, preferredLanguage) {
return targets.some((t) => {
return t.os === preferredPlatform && t.language === preferredLanguage;
});
}

function hasLang(targets, preferredLanguage) {
return targets.some((t) => {
return t.language === preferredLanguage;
});
}

// hasPage is always called after the index is created,
// hence just return the variable in memory.
// There is no need to re-read the index file again.
Expand All @@ -53,7 +115,9 @@ function commandsFor(platform) {
.then((idx) => {
let commands = Object.keys(idx)
.filter((cmd) => {
return idx[cmd].indexOf(platform) !== -1 || idx[cmd].indexOf('common') !== -1;
let targets = idx[cmd].targets;
let platforms = targets.map((t) => {return t.os;});
return platforms.indexOf(platform) !== -1 || platforms.indexOf('common') !== -1;
})
.sort();
return commands;
Expand Down Expand Up @@ -125,11 +189,24 @@ function buildShortPagesIndex() {
let reducer = (index, file) => {
let os = utils.parsePlatform(file);
let page = utils.parsePagename(file);
let language = utils.parseLanguage(file);
if (index[page]) {
index[page].push(os);
let targets = index[page].targets;
let needsPush = true;
for (const target of targets) {
if (target.platform === os && target.language === language) {
needsPush = false;
continue;
}
}
if (needsPush) {
targets.push({'os': os, 'language': language});
index[page].targets = targets;
}
} else {
index[page] = [os];
index[page] = {targets: [{'os': os, 'language': language}]};
}

return index;
};
return files.reduce(reducer, {});
Expand All @@ -142,7 +219,7 @@ function buildShortPagesIndex() {
module.exports = {
getShortIndex,
hasPage,
findPlatform,
findPage,
commands,
commandsFor,
clearPagesIndex,
Expand Down
2 changes: 1 addition & 1 deletion lib/search.js
Original file line number Diff line number Diff line change
Expand Up @@ -197,7 +197,7 @@ exports.printResults = (results, config) => {
results.forEach((elem) => {
let cmdname = utils.parsePagename(elem.file);
let output = ' $ ' + cmdname;
let array = shortIndex[cmdname];
let array = shortIndex[cmdname]['platforms'];
if (array.indexOf('common') === -1 && array.indexOf(preferredPlatform) === -1) {
output += ' (Available on: ' + array.join(', ') + ')';
}
Expand Down
15 changes: 13 additions & 2 deletions lib/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,24 @@ const flatten = require('lodash/flatten');

module.exports = {
parsePlatform(pagefile) {
return path.dirname(pagefile);
const components = pagefile.split(path.sep);
return components[components.length-2];
},

parsePagename(pagefile) {
return path.basename(pagefile, '.md');
},

parseLanguage(pagefile) {
const components = pagefile.split(path.sep);
const langPathIndex = 3;
const langParts = components[components.length-langPathIndex].split('.');
if (langParts.length === 1) {
return 'en';
}
return langParts[1];
},

isPage(file) {
return path.extname(file) === '.md';
},
Expand All @@ -34,7 +45,7 @@ module.exports = {
if (stat.isDirectory()) {
return walk(itemPath);
}
return path.join(path.basename(dir), item);
return path.join(dir, item);
});
}));
})
Expand Down
12 changes: 6 additions & 6 deletions test/cache.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -95,44 +95,44 @@ describe('Cache', () => {
it('should return page contents for ls', () => {
sinon.stub(fs, 'readFile').resolves('# ls\n> ls page');
sinon.stub(platform, 'getPreferredPlatformFolder').returns('osx');
sinon.stub(index, 'findPlatform').resolves('osx');
sinon.stub(index, 'findPage').resolves('osx');
const cache = new Cache(config.get());
return cache.getPage('ls')
.then((content) => {
should.exist(content);
content.should.startWith('# ls');
fs.readFile.restore();
platform.getPreferredPlatformFolder.restore();
index.findPlatform.restore();
index.findPage.restore();
});
});

it('should return empty contents for svcs on OSX', () =>{
sinon.stub(fs, 'readFile').resolves('# svcs\n> svcs');
sinon.stub(platform, 'getPreferredPlatformFolder').returns('osx');
sinon.stub(index, 'findPlatform').resolves(null);
sinon.stub(index, 'findPage').resolves(null);
const cache = new Cache(config.get());
return cache.getPage('svc')
.then((content) => {
should.not.exist(content);
fs.readFile.restore();
platform.getPreferredPlatformFolder.restore();
index.findPlatform.restore();
index.findPage.restore();
});
});

it('should return page contents for svcs on SunOS', () => {
sinon.stub(fs, 'readFile').resolves('# svcs\n> svcs');
sinon.stub(platform, 'getPreferredPlatformFolder').returns('sunos');
sinon.stub(index, 'findPlatform').resolves('svcs');
sinon.stub(index, 'findPage').resolves('svcs');
const cache = new Cache(config.get());
return cache.getPage('svcs')
.then((content) => {
should.exist(content);
content.should.startWith('# svcs');
fs.readFile.restore();
platform.getPreferredPlatformFolder.restore();
index.findPlatform.restore();
index.findPage.restore();
});
});

Expand Down
8 changes: 5 additions & 3 deletions test/functional-test.sh
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,15 @@ function tldr-render-pages {
tldr du --os=osx && \
tldr du --os=linux --markdown && \
tldr du --os=osx --markdown && \
tldr --random && \
tldr --random-example && \
LANG= tldr --random && \
LANG= tldr --random-example && \
tldr --list && \
tldr --list-all
}

tldr --render $HOME/.tldr/cache/pages/common/ssh.md && \
tldr --update && tldr-render-pages && \
tldr --clear-cache && \
tldr --update && tldr-render-pages
tldr --update && tldr-render-pages && \
LANG=pt_BR tldr-render-pages && \
tldr --search "disk space"
Loading

0 comments on commit f2b1085

Please sign in to comment.