Name | cdocs JSON |
Version |
0.0.42
JSON |
| download |
home_page | |
Summary | |
upload_time | 2024-03-02 20:46:44 |
maintainer | |
docs_url | None |
author | David Kershaw |
requires_python | >=3.12,<4.0 |
license | |
keywords |
|
VCS |
|
bugtrack_url |
|
requirements |
No requirements were recorded.
|
Travis-CI |
No Travis.
|
coveralls test coverage |
No coveralls.
|
# Cdocs
## A super simple contextual help library.
**<a href='https://pypi.org/project/cdocs/'>Cdocs</a>** is a Python library that is intended to help manage files for use in <a href='https://en.wikipedia.org/wiki/Context-sensitive_help'>context sensitive help</a> and <a href='https://en.wikipedia.org/wiki/Internationalization_and_localization'>I18n</a>. The library knows how to find docs given a path called a docpath. Docpaths should mirror a logical or physical structure of the application. This way docs are easy find and map cleanly to the way the app works.
<img width="75%" height="75%" src="https://raw.githubusercontent.com/dk107dk/cdocs/master/resources/images/docpath.png"/>
The expected use of Cdocs is to create an endpoint to retrieve in-context help docs and/or UI labels from at runtime. While Cdocs is not an API, the <a href='https://docs.readthedocs.io/en/stable/guides/embedding-content.html'>ReadTheDocs Embed API</a> is a good comparison point. Cdocs could also be used to pull docs into an app build. Cdocs is not is not a <a href='https://en.wikipedia.org/wiki/Headless_content_management_system'>headless CMS</a> but it could be used to create one.
Cdocs stores documentation as text files in a directory tree. Files are <a href='https://jinja.palletsprojects.com/en/2.11.x/'>Jinja templates</a> containing whatever doc format you prefer. Cdocs applies tokens to the templates if ```tokens.json``` containing dict are found. Files can be concatenated and/or composed.
<img width="75%" height="75%" src="https://raw.githubusercontent.com/dk107dk/cdocs/master/resources/images/json-xml-html.png"/>
Cdocs can also be used to find text strings referred to as labels. Labels are stored as (non-template) json files containing dict in the same directory tree. The expected use is to find a JSON structure of labels for UI elements that would be seen in the same screen contextual help is available for.
Requests for labels and tokens are aggregated from every directory below root, starting from the requested path. The values found lowest in the directory tree win.
You may configure as many docs directory trees as you find useful. Multi tree requests go against a Context object that implements the MultiContextDocs abstract base class and returns docs by searching the trees in the order they appear in config.ini. You may limit each directory tree to certain file types. Each root is self-contained, meaning that labels and tokens used in doc file transformations are found and applied only within their directory tree. However, multi-context requests for labels return labels aggregated from all trees. If this is not desireable, use the MultiContextDocs interface passing in only the roots you want labels from.
Every transform template -- a doc found at a docpath -- has access to the MultiContextDocs methods and the ContextualDocs methods. The template also includes a ```docroot``` variable that indicates the root of the cdocs object that is doing the transformation. This means that transformed docs can include pointers to specific roots to resolve included docpath. For example:
```{{ get_doc_from_roots(['public', 'en_US', 'en_UK'], '/app/home/teams/todos/assignee#edit_assignee') }}```
One use for this might be for one root to hold user facing docs composed of reusable chunks of content from another root. Finding translated help docs from the same locale would be another good use for this.
Cdocs will return binary files based on a limited set of known file types. Currently the types are 'gif', 'jpeg', 'jpg', 'png', 'pdf'.
Use a *config.ini* to set the root directories, file extension, etc. The default location for the config file is ```os.getcwd()/config/config.ini```. You can pass a different path in the Cdocs constructor. The contents should be similar to this:
```
# docs section lists the doc directory roots. use as many as you need.
[docs]
public = /Users/dk/dev/cdocs/docs/example
internal = /Users/dk/dev/cdocs/docs/internal
# accepts limits the file types handled by each root. root names passed to the context will be filtered according to this list.
[accepts]
public = cdocs
internal = gif,png,jpeg
# you can configure defaults to point to a file to return if a doc is not found. if not configured, the response to a path that doesn't exist is None.
[defaults]
notfound = internal/404
# format section indicates the extensions found using /x/y/z paths. 'ext' is the default. others are named by their roots. the formats and accepts sections should agree, excepting where accepts is 'cdocs'
[formats]
ext = xml
internal = gif,png,jpeg
# filenames section lets you change the names of the tokens and labels files, set the name anchor char (hashmark), and the on the fly concatenation char (plus)
[filenames]
tokens = tokens.json
labels = labels.json
hashmark = #
plus = +
```
Cdocs has several ways of getting content:
- **get_doc**: docs at paths like ```/x/y/z``` with the physical file being at ```[root]/x/y/z.[ext]```. This is the "default" doc for the path.
The resulting text will be treated as a jinja template. The template will receive a dict created from all the *[tokens].json* files from the doc's directory up to the first level below root, and from the same starting point under the internal tree. The tokens dict will also have all the labels from the ```labels.json``` files found by crawling up the directory trees from the same point. Labels are added to the tokens dict with keys like ```label__[label_key]```.
Templates can use ```plural```, ```cap``` and ```article``` to transform words. These functions come from the <a href='https://pypi.org/project/inflect/'>Inflect library</a>. Let's say there is a a token in the JSON found by aggregating tokens.json files that is named ```elephant``` and is equal to the string "elephant". Examples: ```{{plural(elephant)}}``` would result in *"elephants"*. ```{{cap(elephant)}}``` would result in *"Elephant"*. And ```{{article(elephant)}}``` would resut in *"an elephant"*.
Docs can be incorporated in other docs using jinja expressions like: ```{{ get_doc('/app/home/teams/todos/assignee#edit_assignee') }}```.
This functionality is essentially the same as the more specific *get_compose_doc* method, below.
Paths may not have periods in them.
You may use '+' (or the config value at [filenames][plus]) to concat docs from the same path on the fly. For e.g.
```/x/y/z#name+new_name+edit_name``` would return the doc at ```/x/y/z/name.xml``` + ```/x/y/z/new_name.xml``` + ```/x/y/z/edit_name.xml```, with a space char joining them. If you configure a JSON root and request ```/x/y/z+pdq``` the resulting JSON dict will be the union of ```/x/y/z.json``` and ```/x/y/z/pdq.json```.
- **get_doc**: docs at paths like ```/x/y/z#name``` found as ```[root]/x/y/z/name.[ext]```. These docs are processed in the same way as the default doc. The name separator can be configured to be a different character by adding a [hashmark] value to the config ini file under [filenames].
- **get_concat_doc**: docs as concats of files for paths like ```/x/y/z``` found as ```[root]/x/y/z/page.concat``` where page.concat is a list of simple doc names to be concatenated. Simple doc names are the same as docs named by the *#name* path suffix. The files to be concatenated may be anywhere in the document tree. As noted above, JSON docs that are concatted will result in a single JSON dict structure that is the union of the JSON in the individual docs.
- **get_compose_doc**: docs as jinja files at paths like ```/x/y/z/page.html``` that compose pages where docs are pulled in using jinja expressions like:
```{{ get_doc('/app/home/teams/todos/assignee#edit_assignee') }}```.
*get_compose_doc* requires the compose template be *.xml*, *.html* or *.md*. A compose doc could be referenced by a concat file, or vice versa, but the reference will only include the file contents; it will not be transformed.
- **get_labels**: labels as json dicts for paths like ```/x/y/z``` found as ```[root]/x/y/z/labels.json```. Labels are transformed in the same way as docs, except that the keys of the json dict are individual templates. A label can pull in a doc in the same way that a doc is embedded in a compose doc. Labels that pull in docs are tricky because the docs pulled in may or may not correctly handle any label replacements used with tokens.json, depending on any circular references.
- **list_docs**: returns a list of simple names of docs below the docpath. This method will return [```/x/y/z/a.xml```, ```/x/y/z/b.xml```] (actually [```'a.xml'```,```'b.xml'```] for ```/x/y/z``` but it won't include ```/x/y/z.xml``` (or, actually, the simple name ```'z.xml'```).
<img width="75%" height="75%" src="https://raw.githubusercontent.com/dk107dk/cdocs/master/resources/images/labels.png"/>
Each of these retrieval methods is matched by a similarly named "..._from_roots" version that takes a List of root names. See the MultiContextDocs abstract base class.
Creating a Cdocs endpoint using <a href='https://flask.palletsprojects.com/en/1.1.x/'>Flask</a> might look something like:
```
app = Flask(__name__)
api = Api(app)
@app.route('/cdocs/<path:cdocspath>')
def cdocs(cdocspath:str):
cdocs = Cdocs(PATH)
return cdocs.get_doc(cdocspath)
```
Flask won't have access to an anchor appended to a URL by a hashmark. (E.g. ```http://a.com/x/y/z#name```). To work around that you can add a [filenames][hashmark] setting to the config.ini with a different character. E.g.:
```
[filenames]
hashmark = *
```
The same endpoint using all of a set of docs directory trees configured in config.ini might look like:
```
app = Flask(__name__)
api = Api(app)
@app.route('/cdocs/<path:cdocspath>')
def cdocs(cdocspath:str):
metadata = ContextMetaData()
context = Context(metadata)
return context.get_doc(cdocspath)
```
An endpoint implementation might want to search certain trees based on the locale of the request, a version name, a product name, an author, etc.
An SPA might use the endpoint to pull docs or labels for context sensitive help, UI labels, internationalization, etc. by converting its router's current route to docpath. A route like ```/teams/124/projects/15/todos``` might be converted to ```/teams/projects/todos``` to get UI labels and tooltips using something like:
```
getDocpathFromRoute(route) {
let parts = route.split("/")
var path = ""
for (let i = 0; i <parts.length; i++) {
if (isNaN(Number(parts[i]))) {
path += "/" + parts[i]
}
}
return path
}
```
Cdocs is on Pypi <a href='https://pypi.org/project/cdocs/'>here</a>.
Raw data
{
"_id": null,
"home_page": "",
"name": "cdocs",
"maintainer": "",
"docs_url": null,
"requires_python": ">=3.12,<4.0",
"maintainer_email": "",
"keywords": "",
"author": "David Kershaw",
"author_email": "dk107dk@hotmail.com",
"download_url": "https://files.pythonhosted.org/packages/ac/43/3141b60551b16326e1a313638dc85b677d93a6c2a5aaea95e6243ecb8c1c/cdocs-0.0.42.tar.gz",
"platform": null,
"description": "# Cdocs\n## A super simple contextual help library.\n\n**<a href='https://pypi.org/project/cdocs/'>Cdocs</a>** is a Python library that is intended to help manage files for use in <a href='https://en.wikipedia.org/wiki/Context-sensitive_help'>context sensitive help</a> and <a href='https://en.wikipedia.org/wiki/Internationalization_and_localization'>I18n</a>. The library knows how to find docs given a path called a docpath. Docpaths should mirror a logical or physical structure of the application. This way docs are easy find and map cleanly to the way the app works.\n\n<img width=\"75%\" height=\"75%\" src=\"https://raw.githubusercontent.com/dk107dk/cdocs/master/resources/images/docpath.png\"/>\n\nThe expected use of Cdocs is to create an endpoint to retrieve in-context help docs and/or UI labels from at runtime. While Cdocs is not an API, the <a href='https://docs.readthedocs.io/en/stable/guides/embedding-content.html'>ReadTheDocs Embed API</a> is a good comparison point. Cdocs could also be used to pull docs into an app build. Cdocs is not is not a <a href='https://en.wikipedia.org/wiki/Headless_content_management_system'>headless CMS</a> but it could be used to create one.\n\nCdocs stores documentation as text files in a directory tree. Files are <a href='https://jinja.palletsprojects.com/en/2.11.x/'>Jinja templates</a> containing whatever doc format you prefer. Cdocs applies tokens to the templates if ```tokens.json``` containing dict are found. Files can be concatenated and/or composed.\n\n<img width=\"75%\" height=\"75%\" src=\"https://raw.githubusercontent.com/dk107dk/cdocs/master/resources/images/json-xml-html.png\"/>\n\nCdocs can also be used to find text strings referred to as labels. Labels are stored as (non-template) json files containing dict in the same directory tree. The expected use is to find a JSON structure of labels for UI elements that would be seen in the same screen contextual help is available for.\n\nRequests for labels and tokens are aggregated from every directory below root, starting from the requested path. The values found lowest in the directory tree win.\n\nYou may configure as many docs directory trees as you find useful. Multi tree requests go against a Context object that implements the MultiContextDocs abstract base class and returns docs by searching the trees in the order they appear in config.ini. You may limit each directory tree to certain file types. Each root is self-contained, meaning that labels and tokens used in doc file transformations are found and applied only within their directory tree. However, multi-context requests for labels return labels aggregated from all trees. If this is not desireable, use the MultiContextDocs interface passing in only the roots you want labels from.\n\nEvery transform template -- a doc found at a docpath -- has access to the MultiContextDocs methods and the ContextualDocs methods. The template also includes a ```docroot``` variable that indicates the root of the cdocs object that is doing the transformation. This means that transformed docs can include pointers to specific roots to resolve included docpath. For example:\n\n```{{ get_doc_from_roots(['public', 'en_US', 'en_UK'], '/app/home/teams/todos/assignee#edit_assignee') }}```\n\nOne use for this might be for one root to hold user facing docs composed of reusable chunks of content from another root. Finding translated help docs from the same locale would be another good use for this.\n\nCdocs will return binary files based on a limited set of known file types. Currently the types are 'gif', 'jpeg', 'jpg', 'png', 'pdf'.\n\nUse a *config.ini* to set the root directories, file extension, etc. The default location for the config file is ```os.getcwd()/config/config.ini```. You can pass a different path in the Cdocs constructor. The contents should be similar to this:\n```\n# docs section lists the doc directory roots. use as many as you need.\n[docs]\n public = /Users/dk/dev/cdocs/docs/example\n internal = /Users/dk/dev/cdocs/docs/internal\n# accepts limits the file types handled by each root. root names passed to the context will be filtered according to this list.\n[accepts]\n public = cdocs\n internal = gif,png,jpeg\n# you can configure defaults to point to a file to return if a doc is not found. if not configured, the response to a path that doesn't exist is None.\n[defaults]\n notfound = internal/404\n# format section indicates the extensions found using /x/y/z paths. 'ext' is the default. others are named by their roots. the formats and accepts sections should agree, excepting where accepts is 'cdocs'\n[formats]\n ext = xml\n internal = gif,png,jpeg\n# filenames section lets you change the names of the tokens and labels files, set the name anchor char (hashmark), and the on the fly concatenation char (plus)\n[filenames]\n tokens = tokens.json\n labels = labels.json\n hashmark = #\n plus = +\n```\n\nCdocs has several ways of getting content:\n - **get_doc**: docs at paths like ```/x/y/z``` with the physical file being at ```[root]/x/y/z.[ext]```. This is the \"default\" doc for the path.\n\n The resulting text will be treated as a jinja template. The template will receive a dict created from all the *[tokens].json* files from the doc's directory up to the first level below root, and from the same starting point under the internal tree. The tokens dict will also have all the labels from the ```labels.json``` files found by crawling up the directory trees from the same point. Labels are added to the tokens dict with keys like ```label__[label_key]```.\n\n Templates can use ```plural```, ```cap``` and ```article``` to transform words. These functions come from the <a href='https://pypi.org/project/inflect/'>Inflect library</a>. Let's say there is a a token in the JSON found by aggregating tokens.json files that is named ```elephant``` and is equal to the string \"elephant\". Examples: ```{{plural(elephant)}}``` would result in *\"elephants\"*. ```{{cap(elephant)}}``` would result in *\"Elephant\"*. And ```{{article(elephant)}}``` would resut in *\"an elephant\"*.\n\n Docs can be incorporated in other docs using jinja expressions like: ```{{ get_doc('/app/home/teams/todos/assignee#edit_assignee') }}```.\n This functionality is essentially the same as the more specific *get_compose_doc* method, below.\n\n Paths may not have periods in them.\n\n You may use '+' (or the config value at [filenames][plus]) to concat docs from the same path on the fly. For e.g.\n```/x/y/z#name+new_name+edit_name``` would return the doc at ```/x/y/z/name.xml``` + ```/x/y/z/new_name.xml``` + ```/x/y/z/edit_name.xml```, with a space char joining them. If you configure a JSON root and request ```/x/y/z+pdq``` the resulting JSON dict will be the union of ```/x/y/z.json``` and ```/x/y/z/pdq.json```.\n - **get_doc**: docs at paths like ```/x/y/z#name``` found as ```[root]/x/y/z/name.[ext]```. These docs are processed in the same way as the default doc. The name separator can be configured to be a different character by adding a [hashmark] value to the config ini file under [filenames].\n - **get_concat_doc**: docs as concats of files for paths like ```/x/y/z``` found as ```[root]/x/y/z/page.concat``` where page.concat is a list of simple doc names to be concatenated. Simple doc names are the same as docs named by the *#name* path suffix. The files to be concatenated may be anywhere in the document tree. As noted above, JSON docs that are concatted will result in a single JSON dict structure that is the union of the JSON in the individual docs.\n - **get_compose_doc**: docs as jinja files at paths like ```/x/y/z/page.html``` that compose pages where docs are pulled in using jinja expressions like:\n```{{ get_doc('/app/home/teams/todos/assignee#edit_assignee') }}```.\n*get_compose_doc* requires the compose template be *.xml*, *.html* or *.md*. A compose doc could be referenced by a concat file, or vice versa, but the reference will only include the file contents; it will not be transformed.\n - **get_labels**: labels as json dicts for paths like ```/x/y/z``` found as ```[root]/x/y/z/labels.json```. Labels are transformed in the same way as docs, except that the keys of the json dict are individual templates. A label can pull in a doc in the same way that a doc is embedded in a compose doc. Labels that pull in docs are tricky because the docs pulled in may or may not correctly handle any label replacements used with tokens.json, depending on any circular references.\n - **list_docs**: returns a list of simple names of docs below the docpath. This method will return [```/x/y/z/a.xml```, ```/x/y/z/b.xml```] (actually [```'a.xml'```,```'b.xml'```] for ```/x/y/z``` but it won't include ```/x/y/z.xml``` (or, actually, the simple name ```'z.xml'```).\n\n<img width=\"75%\" height=\"75%\" src=\"https://raw.githubusercontent.com/dk107dk/cdocs/master/resources/images/labels.png\"/>\n\nEach of these retrieval methods is matched by a similarly named \"..._from_roots\" version that takes a List of root names. See the MultiContextDocs abstract base class.\n\nCreating a Cdocs endpoint using <a href='https://flask.palletsprojects.com/en/1.1.x/'>Flask</a> might look something like:\n```\napp = Flask(__name__)\napi = Api(app)\n@app.route('/cdocs/<path:cdocspath>')\ndef cdocs(cdocspath:str):\n cdocs = Cdocs(PATH)\n return cdocs.get_doc(cdocspath)\n```\n\nFlask won't have access to an anchor appended to a URL by a hashmark. (E.g. ```http://a.com/x/y/z#name```). To work around that you can add a [filenames][hashmark] setting to the config.ini with a different character. E.g.:\n```\n[filenames]\n hashmark = *\n```\n\nThe same endpoint using all of a set of docs directory trees configured in config.ini might look like:\n```\napp = Flask(__name__)\napi = Api(app)\n@app.route('/cdocs/<path:cdocspath>')\ndef cdocs(cdocspath:str):\n metadata = ContextMetaData()\n context = Context(metadata)\n return context.get_doc(cdocspath)\n```\nAn endpoint implementation might want to search certain trees based on the locale of the request, a version name, a product name, an author, etc.\n\nAn SPA might use the endpoint to pull docs or labels for context sensitive help, UI labels, internationalization, etc. by converting its router's current route to docpath. A route like ```/teams/124/projects/15/todos``` might be converted to ```/teams/projects/todos``` to get UI labels and tooltips using something like:\n```\n getDocpathFromRoute(route) {\n let parts = route.split(\"/\")\n var path = \"\"\n for (let i = 0; i <parts.length; i++) {\n if (isNaN(Number(parts[i]))) {\n path += \"/\" + parts[i]\n }\n }\n return path\n }\n```\n\n\nCdocs is on Pypi <a href='https://pypi.org/project/cdocs/'>here</a>.\n\n\n\n",
"bugtrack_url": null,
"license": "",
"summary": "",
"version": "0.0.42",
"project_urls": null,
"split_keywords": [],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "49731545d85f30d641593f29d95bfefb503844506934466c8bf61389febe2621",
"md5": "2fc8151f9a26b37188df83d249b527db",
"sha256": "9818028868521cf7894d107ff73b3273b9be2cf95d05394dceefd1d2da17b285"
},
"downloads": -1,
"filename": "cdocs-0.0.42-py3-none-any.whl",
"has_sig": false,
"md5_digest": "2fc8151f9a26b37188df83d249b527db",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.12,<4.0",
"size": 28831,
"upload_time": "2024-03-02T20:46:41",
"upload_time_iso_8601": "2024-03-02T20:46:41.530717Z",
"url": "https://files.pythonhosted.org/packages/49/73/1545d85f30d641593f29d95bfefb503844506934466c8bf61389febe2621/cdocs-0.0.42-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "ac433141b60551b16326e1a313638dc85b677d93a6c2a5aaea95e6243ecb8c1c",
"md5": "d3976f7a7f30b2af3d5b09bfffa75e48",
"sha256": "ea138f4fcb36947d3d4757f6153770f055d86c616f1a66fe9e72d7993501be74"
},
"downloads": -1,
"filename": "cdocs-0.0.42.tar.gz",
"has_sig": false,
"md5_digest": "d3976f7a7f30b2af3d5b09bfffa75e48",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.12,<4.0",
"size": 25890,
"upload_time": "2024-03-02T20:46:44",
"upload_time_iso_8601": "2024-03-02T20:46:44.022815Z",
"url": "https://files.pythonhosted.org/packages/ac/43/3141b60551b16326e1a313638dc85b677d93a6c2a5aaea95e6243ecb8c1c/cdocs-0.0.42.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-03-02 20:46:44",
"github": false,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"lcname": "cdocs"
}