cmem-cmemc 24.1.1rc1__tar.gz → 24.1.3__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- cmem_cmemc-24.1.3/PKG-INFO +87 -0
- cmem_cmemc-24.1.3/README-public.md +36 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/__init__.py +4 -9
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/__init__.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/admin.py +4 -6
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/config.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/dataset.py +2 -1
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/graph.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/project.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/python.py +5 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/query.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/resource.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/scheduler.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/store.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/user.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/validation.py +7 -6
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/variable.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/vocabulary.py +11 -10
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/workflow.py +6 -5
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/workspace.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/completion.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/constants.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/context.py +16 -10
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/manual_helper/graph.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/manual_helper/multi_page.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/object_list.py +1 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/utils.py +21 -2
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/pyproject.toml +23 -4
- cmem_cmemc-24.1.1rc1/PKG-INFO +0 -68
- cmem_cmemc-24.1.1rc1/README-public.md +0 -23
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/LICENSE +0 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/_cmemc.zsh +0 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/acl.py +0 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/client.py +0 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/commands/metrics.py +0 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/exceptions.py +0 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/manual_helper/__init__.py +0 -0
- {cmem_cmemc-24.1.1rc1 → cmem_cmemc-24.1.3}/cmem_cmemc/manual_helper/single_page.py +0 -0
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
Metadata-Version: 2.1
|
|
2
|
+
Name: cmem-cmemc
|
|
3
|
+
Version: 24.1.3
|
|
4
|
+
Summary: Command line client for eccenca Corporate Memory
|
|
5
|
+
Home-page: https://eccenca.com/go/cmemc
|
|
6
|
+
License: Apache 2.0
|
|
7
|
+
Author: eccenca
|
|
8
|
+
Author-email: cmempy-developer@eccenca.com
|
|
9
|
+
Requires-Python: >=3.9,<4.0
|
|
10
|
+
Classifier: Development Status :: 5 - Production/Stable
|
|
11
|
+
Classifier: Environment :: Console
|
|
12
|
+
Classifier: Intended Audience :: Customer Service
|
|
13
|
+
Classifier: Intended Audience :: Developers
|
|
14
|
+
Classifier: Intended Audience :: Information Technology
|
|
15
|
+
Classifier: Intended Audience :: Science/Research
|
|
16
|
+
Classifier: Intended Audience :: System Administrators
|
|
17
|
+
Classifier: License :: OSI Approved :: Apache Software License
|
|
18
|
+
Classifier: License :: Other/Proprietary License
|
|
19
|
+
Classifier: Natural Language :: English
|
|
20
|
+
Classifier: Operating System :: OS Independent
|
|
21
|
+
Classifier: Programming Language :: Python :: 3
|
|
22
|
+
Classifier: Programming Language :: Python :: 3.9
|
|
23
|
+
Classifier: Programming Language :: Python :: 3.10
|
|
24
|
+
Classifier: Programming Language :: Python :: 3.11
|
|
25
|
+
Classifier: Programming Language :: Python :: 3.12
|
|
26
|
+
Classifier: Programming Language :: Python :: 3 :: Only
|
|
27
|
+
Classifier: Topic :: Database
|
|
28
|
+
Classifier: Topic :: Software Development :: Testing
|
|
29
|
+
Classifier: Topic :: Utilities
|
|
30
|
+
Requires-Dist: beautifulsoup4 (>=4.12.2,<5.0.0)
|
|
31
|
+
Requires-Dist: certifi (>=2023.5.7)
|
|
32
|
+
Requires-Dist: click (>=8.1.7,<9.0.0)
|
|
33
|
+
Requires-Dist: click-didyoumean (>=0.3.0,<0.4.0)
|
|
34
|
+
Requires-Dist: click-help-colors (>=0.9.1,<0.10.0)
|
|
35
|
+
Requires-Dist: cmem-cmempy (==24.1.1)
|
|
36
|
+
Requires-Dist: configparser (>=5.3.0,<6.0.0)
|
|
37
|
+
Requires-Dist: jinja2 (>=3.1.2,<4.0.0)
|
|
38
|
+
Requires-Dist: natsort (>=8.3.1,<9.0.0)
|
|
39
|
+
Requires-Dist: prometheus-client (>=0.16.0,<0.17.0)
|
|
40
|
+
Requires-Dist: pygments (>=2.15.1,<3.0.0)
|
|
41
|
+
Requires-Dist: pyjwt (>=2.7.0,<3.0.0)
|
|
42
|
+
Requires-Dist: python-dateutil (>=2.8.2,<3.0.0)
|
|
43
|
+
Requires-Dist: requests (>=2.31.0,<3.0.0)
|
|
44
|
+
Requires-Dist: rich (>=13.7.0,<14.0.0)
|
|
45
|
+
Requires-Dist: six (>=1.16.0,<2.0.0)
|
|
46
|
+
Requires-Dist: timeago (>=1.0.16,<2.0.0)
|
|
47
|
+
Requires-Dist: treelib (>=1.6.4,<2.0.0)
|
|
48
|
+
Requires-Dist: urllib3 (>=2,<3)
|
|
49
|
+
Description-Content-Type: text/markdown
|
|
50
|
+
|
|
51
|
+
# cmemc
|
|
52
|
+
|
|
53
|
+
cmemc is the official command line client for [eccenca Corporate Memory](https://documentation.eccenca.com/).
|
|
54
|
+
|
|
55
|
+
[![version][version-shield]][changelog] [![Python Versions][python-shield]][pypi] [![eccenca Corporate Memory][cmem-shield]][cmem]
|
|
56
|
+
|
|
57
|
+
[![teaser][teaser-image]][cmemc-docu]
|
|
58
|
+
|
|
59
|
+
## Installation
|
|
60
|
+
|
|
61
|
+
In order to install the cmemc, run:
|
|
62
|
+
|
|
63
|
+
pipx install cmem-cmemc
|
|
64
|
+
|
|
65
|
+
Of course you can install cmemc also with pip, but we recommend [pipx](https://pypa.github.io/pipx/) for normal desktop usage.
|
|
66
|
+
|
|
67
|
+
## Configuration and Usage
|
|
68
|
+
|
|
69
|
+
cmemc is intended for System Administrators and Linked Data Expert, who wants to automate and remote control activities on eccenca Corporate Memory.
|
|
70
|
+
|
|
71
|
+
The cmemc manual including basic usage pattern, configuration as well as a command reference is available at:
|
|
72
|
+
|
|
73
|
+
[https://eccenca.com/go/cmemc](https://eccenca.com/go/cmemc)
|
|
74
|
+
|
|
75
|
+
cmemc only works with Python 3 and refuses to work with Python 2.x.
|
|
76
|
+
In addition to that, cmemc will warn you in case an untested Python environment is used.
|
|
77
|
+
|
|
78
|
+
|
|
79
|
+
[version-shield]: https://badge.fury.io/py/cmem-cmemc.svg
|
|
80
|
+
[changelog]: https://pypi.org/project/cmem-cmemc/#history
|
|
81
|
+
[python-shield]: https://img.shields.io/pypi/pyversions/cmem-cmemc.svg
|
|
82
|
+
[pypi]: https://pypi.org/project/cmem-cmemc/
|
|
83
|
+
[cmem]: https://documentation.eccenca.com
|
|
84
|
+
[cmem-shield]: https://img.shields.io/badge/made%20for-eccenca%20Corporate%20Memory-blue?logo=
|
|
85
|
+
[teaser-image]: https://documentation.eccenca.com/24.1/automate/cmemc-command-line-interface/configuration/completion-setup/22.1-cmemc-create-dataset.gif
|
|
86
|
+
[cmemc-docu]: https://eccenca.com/go/cmemc
|
|
87
|
+
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
# cmemc
|
|
2
|
+
|
|
3
|
+
cmemc is the official command line client for [eccenca Corporate Memory](https://documentation.eccenca.com/).
|
|
4
|
+
|
|
5
|
+
[![version][version-shield]][changelog] [![Python Versions][python-shield]][pypi] [![eccenca Corporate Memory][cmem-shield]][cmem]
|
|
6
|
+
|
|
7
|
+
[![teaser][teaser-image]][cmemc-docu]
|
|
8
|
+
|
|
9
|
+
## Installation
|
|
10
|
+
|
|
11
|
+
In order to install the cmemc, run:
|
|
12
|
+
|
|
13
|
+
pipx install cmem-cmemc
|
|
14
|
+
|
|
15
|
+
Of course you can install cmemc also with pip, but we recommend [pipx](https://pypa.github.io/pipx/) for normal desktop usage.
|
|
16
|
+
|
|
17
|
+
## Configuration and Usage
|
|
18
|
+
|
|
19
|
+
cmemc is intended for System Administrators and Linked Data Expert, who wants to automate and remote control activities on eccenca Corporate Memory.
|
|
20
|
+
|
|
21
|
+
The cmemc manual including basic usage pattern, configuration as well as a command reference is available at:
|
|
22
|
+
|
|
23
|
+
[https://eccenca.com/go/cmemc](https://eccenca.com/go/cmemc)
|
|
24
|
+
|
|
25
|
+
cmemc only works with Python 3 and refuses to work with Python 2.x.
|
|
26
|
+
In addition to that, cmemc will warn you in case an untested Python environment is used.
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
[version-shield]: https://badge.fury.io/py/cmem-cmemc.svg
|
|
30
|
+
[changelog]: https://pypi.org/project/cmem-cmemc/#history
|
|
31
|
+
[python-shield]: https://img.shields.io/pypi/pyversions/cmem-cmemc.svg
|
|
32
|
+
[pypi]: https://pypi.org/project/cmem-cmemc/
|
|
33
|
+
[cmem]: https://documentation.eccenca.com
|
|
34
|
+
[cmem-shield]: https://img.shields.io/badge/made%20for-eccenca%20Corporate%20Memory-blue?logo=
|
|
35
|
+
[teaser-image]: https://documentation.eccenca.com/24.1/automate/cmemc-command-line-interface/configuration/completion-setup/22.1-cmemc-create-dataset.gif
|
|
36
|
+
[cmemc-docu]: https://eccenca.com/go/cmemc
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
"""The main command line interface."""
|
|
2
|
+
|
|
2
3
|
import contextlib
|
|
3
4
|
import os
|
|
4
5
|
import sys
|
|
@@ -26,7 +27,7 @@ from cmem_cmemc.exceptions import InvalidConfigurationError
|
|
|
26
27
|
from cmem_cmemc.manual_helper.graph import print_manual_graph
|
|
27
28
|
from cmem_cmemc.manual_helper.multi_page import create_multi_page_documentation
|
|
28
29
|
from cmem_cmemc.manual_helper.single_page import print_manual
|
|
29
|
-
from cmem_cmemc.utils import extract_error_message, get_version
|
|
30
|
+
from cmem_cmemc.utils import check_python_version, extract_error_message, get_version
|
|
30
31
|
|
|
31
32
|
CMEMC_VERSION = get_version()
|
|
32
33
|
|
|
@@ -38,16 +39,10 @@ if os.environ.get("_CMEMC_COMPLETE", "") == "zsh_source":
|
|
|
38
39
|
|
|
39
40
|
version = sys.version_info
|
|
40
41
|
PYTHON_VERSION = f"{version.major}.{version.minor}.{version.micro}"
|
|
41
|
-
|
|
42
|
-
PYTHON_GOT = f"{version.major}.{version.minor}"
|
|
43
|
-
if PYTHON_EXPECTED != PYTHON_GOT and not CONTEXT.is_completing():
|
|
44
|
-
CONTEXT.echo_warning(
|
|
45
|
-
"Warning: You are running cmemc under a non-tested python "
|
|
46
|
-
f"environment (expected {PYTHON_EXPECTED}, got {PYTHON_GOT})"
|
|
47
|
-
)
|
|
42
|
+
check_python_version(ctx=CONTEXT)
|
|
48
43
|
|
|
49
44
|
# set the user-agent environment for the http request headers
|
|
50
|
-
os.environ["CMEM_USER_AGENT"] = f"cmemc/{CMEMC_VERSION}
|
|
45
|
+
os.environ["CMEM_USER_AGENT"] = f"cmemc/{CMEMC_VERSION} (Python {PYTHON_VERSION})"
|
|
51
46
|
|
|
52
47
|
# https://github.com/pallets/click/blob/master/examples/complex/complex/cli.py
|
|
53
48
|
CONTEXT_SETTINGS = {"auto_envvar_prefix": "CMEMC", "help_option_names": ["-h", "--help"]}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
"""admin commands for cmem command line interface."""
|
|
2
|
-
|
|
2
|
+
|
|
3
|
+
from datetime import datetime, timezone
|
|
3
4
|
|
|
4
5
|
import click
|
|
5
6
|
import jwt
|
|
@@ -41,12 +42,9 @@ def _check_graphdb_license(app: ApplicationContext, data: dict, months: int, exi
|
|
|
41
42
|
# DP < 24.1 has no graph license information here
|
|
42
43
|
return
|
|
43
44
|
expiration_date_str = data["dp"]["info"]["store"]["licenseExpiration"]
|
|
44
|
-
|
|
45
|
-
# DP = 24.1 outputs this for GraphDB free ...¯\_(ツ)_/¯
|
|
46
|
-
return
|
|
47
|
-
expiration_date = datetime.strptime(expiration_date_str, "%Y-%m-%d").astimezone(tz=UTC)
|
|
45
|
+
expiration_date = datetime.strptime(expiration_date_str, "%Y-%m-%d").astimezone(tz=timezone.utc)
|
|
48
46
|
grace_starts = expiration_date - relativedelta(months=months)
|
|
49
|
-
if grace_starts < datetime.now(tz=
|
|
47
|
+
if grace_starts < datetime.now(tz=timezone.utc):
|
|
50
48
|
graphdb_license_end = data["dp"]["info"]["store"]["licenseExpiration"]
|
|
51
49
|
output = f"Your GraphDB license expires on {graphdb_license_end}."
|
|
52
50
|
if exit_1 == "always":
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
"""dataset commands for cmem command line interface."""
|
|
2
|
+
|
|
2
3
|
import json
|
|
3
4
|
import re
|
|
4
5
|
from pathlib import Path
|
|
@@ -284,7 +285,7 @@ def _check_or_set_dataset_type(
|
|
|
284
285
|
|
|
285
286
|
"""
|
|
286
287
|
source = Path(dataset_file).name if dataset_file else ""
|
|
287
|
-
target = parameter_dict
|
|
288
|
+
target = parameter_dict.get("file", "")
|
|
288
289
|
suggestions = (
|
|
289
290
|
(".ttl", "file"),
|
|
290
291
|
(".csv", "csv"),
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
"""DataIntegration python management commands."""
|
|
2
|
+
|
|
2
3
|
import sys
|
|
3
4
|
from dataclasses import asdict
|
|
4
5
|
from re import match
|
|
@@ -49,6 +50,10 @@ def install_command(app: ApplicationContext, package: str) -> None:
|
|
|
49
50
|
.tar.gz file, by uploading a build distribution .whl file, or by
|
|
50
51
|
specifying a package name, i.e., a pip requirement specifier with a
|
|
51
52
|
package name available on pypi.org (e.g. `requests==2.27.1`).
|
|
53
|
+
|
|
54
|
+
Note: The tab-completion of this command lists only public packages from
|
|
55
|
+
pypi.org and not from additional or changed python package repositories you
|
|
56
|
+
may have configured on the server.
|
|
52
57
|
"""
|
|
53
58
|
app.echo_info(f"Install package {package} ... ", nl=False)
|
|
54
59
|
try:
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
"""graph validation command group"""
|
|
2
|
+
|
|
2
3
|
import time
|
|
3
|
-
from datetime import
|
|
4
|
+
from datetime import datetime, timezone
|
|
4
5
|
|
|
5
6
|
import click
|
|
6
7
|
import requests
|
|
@@ -144,8 +145,8 @@ def _get_batch_validation_option(validation_: dict) -> tuple[str, str]:
|
|
|
144
145
|
id_ = validation_["id"]
|
|
145
146
|
state = validation_["state"]
|
|
146
147
|
graph = validation_["contextGraphIri"]
|
|
147
|
-
stamp = datetime.fromtimestamp(validation_["executionStarted"] / 1000, tz=
|
|
148
|
-
time_ago = timeago.format(stamp, datetime.now(tz=
|
|
148
|
+
stamp = datetime.fromtimestamp(validation_["executionStarted"] / 1000, tz=timezone.utc)
|
|
149
|
+
time_ago = timeago.format(stamp, datetime.now(tz=timezone.utc))
|
|
149
150
|
resources = _get_resource_count(validation_)
|
|
150
151
|
violations = _get_violation_count(validation_)
|
|
151
152
|
return (
|
|
@@ -295,7 +296,7 @@ def _get_resource_count(batch_validation: dict) -> str:
|
|
|
295
296
|
|
|
296
297
|
def _get_violation_count(process_data: dict) -> str:
|
|
297
298
|
"""Get violation count from validation report"""
|
|
298
|
-
if process_data.get("executionStarted"
|
|
299
|
+
if process_data.get("executionStarted") is None:
|
|
299
300
|
return "-"
|
|
300
301
|
resources = str(process_data.get("resourcesWithViolationsCount", "0"))
|
|
301
302
|
violations = str(process_data.get("violationsCount", "0"))
|
|
@@ -408,8 +409,8 @@ def list_command(ctx: Context, filter_: tuple[tuple[str, str]], id_only: bool, r
|
|
|
408
409
|
table = []
|
|
409
410
|
for _ in validations:
|
|
410
411
|
if "executionStarted" in _ and _["executionStarted"] is not None:
|
|
411
|
-
stamp = datetime.fromtimestamp(_["executionStarted"] / 1000, tz=
|
|
412
|
-
time_ago = timeago.format(stamp, datetime.now(tz=
|
|
412
|
+
stamp = datetime.fromtimestamp(_["executionStarted"] / 1000, tz=timezone.utc)
|
|
413
|
+
time_ago = timeago.format(stamp, datetime.now(tz=timezone.utc))
|
|
413
414
|
else:
|
|
414
415
|
time_ago = f"{_['state']}"
|
|
415
416
|
row = [
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
"""vocabularies commands for cmem command line interface."""
|
|
2
|
+
|
|
2
3
|
import io
|
|
3
|
-
from datetime import
|
|
4
|
+
from datetime import datetime, timezone
|
|
4
5
|
from re import match
|
|
5
6
|
from urllib.parse import urlparse
|
|
6
7
|
|
|
@@ -140,7 +141,7 @@ def _insert_catalog_entry(iri: str, prefix: str, namespace: str, label: str, lan
|
|
|
140
141
|
iri=iri,
|
|
141
142
|
prefix=prefix,
|
|
142
143
|
namespace=namespace,
|
|
143
|
-
date=datetime.now(tz=
|
|
144
|
+
date=datetime.now(tz=timezone.utc).date(),
|
|
144
145
|
label=label,
|
|
145
146
|
language=language,
|
|
146
147
|
description="vocabulary imported with cmemc",
|
|
@@ -150,7 +151,7 @@ def _insert_catalog_entry(iri: str, prefix: str, namespace: str, label: str, lan
|
|
|
150
151
|
|
|
151
152
|
|
|
152
153
|
def _get_vocabulary_metadata_from_file(
|
|
153
|
-
file: io.
|
|
154
|
+
file: io.BytesIO, namespace_given: bool = False
|
|
154
155
|
) -> dict[str, str]:
|
|
155
156
|
"""Get potential graph iri and prefix/namespace from a turtle file."""
|
|
156
157
|
metadata = {"iri": "", "prefix": "", "namespace": ""}
|
|
@@ -345,19 +346,19 @@ def import_command(
|
|
|
345
346
|
description which correctly uses the `vann:preferredNamespacePrefix` and
|
|
346
347
|
`vann:preferredNamespaceUri` properties.
|
|
347
348
|
"""
|
|
348
|
-
|
|
349
|
+
_buffer = io.BytesIO()
|
|
349
350
|
with click.open_file(file, "rb") as file_handle:
|
|
350
|
-
|
|
351
|
-
|
|
351
|
+
_buffer.write(file_handle.read())
|
|
352
|
+
_buffer.seek(0)
|
|
352
353
|
|
|
353
354
|
# fetch metadata
|
|
354
355
|
if namespace != (None, None):
|
|
355
356
|
_validate_namespace(app, namespace)
|
|
356
|
-
meta_data = _get_vocabulary_metadata_from_file(
|
|
357
|
+
meta_data = _get_vocabulary_metadata_from_file(_buffer, True)
|
|
357
358
|
meta_data["prefix"] = namespace[0] # type: ignore[assignment]
|
|
358
359
|
meta_data["namespace"] = namespace[1] # type: ignore[assignment]
|
|
359
360
|
else:
|
|
360
|
-
meta_data = _get_vocabulary_metadata_from_file(
|
|
361
|
+
meta_data = _get_vocabulary_metadata_from_file(_buffer, False)
|
|
361
362
|
iri = meta_data["iri"]
|
|
362
363
|
|
|
363
364
|
success_message = "done"
|
|
@@ -368,8 +369,8 @@ def import_command(
|
|
|
368
369
|
raise ValueError(f"Proposed graph {iri} does already exist.")
|
|
369
370
|
app.echo_info(f"Import {file} as vocabulary to {iri} ... ", nl=False)
|
|
370
371
|
# upload graph
|
|
371
|
-
|
|
372
|
-
graph_api.post_streamed(iri,
|
|
372
|
+
_buffer.seek(0)
|
|
373
|
+
graph_api.post_streamed(iri, _buffer, replace=True)
|
|
373
374
|
|
|
374
375
|
# resolve label
|
|
375
376
|
resolved_label_object: dict = resolve([iri], graph=iri)[iri]
|
|
@@ -1,8 +1,9 @@
|
|
|
1
1
|
"""workflow commands for cmem command line interface."""
|
|
2
|
+
|
|
2
3
|
import re
|
|
3
4
|
import sys
|
|
4
5
|
import time
|
|
5
|
-
from datetime import
|
|
6
|
+
from datetime import datetime, timezone
|
|
6
7
|
from pathlib import Path
|
|
7
8
|
|
|
8
9
|
import click
|
|
@@ -319,8 +320,8 @@ def _workflow_echo_status(app: ApplicationContext, status: dict) -> None:
|
|
|
319
320
|
canceled only exists sometimes
|
|
320
321
|
"""
|
|
321
322
|
# prepare human friendly relative time
|
|
322
|
-
now = datetime.now(tz=
|
|
323
|
-
stamp = datetime.fromtimestamp(status["lastUpdateTime"] / 1000, tz=
|
|
323
|
+
now = datetime.now(tz=timezone.utc)
|
|
324
|
+
stamp = datetime.fromtimestamp(status["lastUpdateTime"] / 1000, tz=timezone.utc)
|
|
324
325
|
time_ago = timeago.format(stamp, now, "en")
|
|
325
326
|
status_name = status["statusName"]
|
|
326
327
|
status_message = status["message"]
|
|
@@ -338,9 +339,9 @@ def _workflow_echo_status(app: ApplicationContext, status: dict) -> None:
|
|
|
338
339
|
f"statusName is {status_name}, expecting one of: " "Running, Canceling or Waiting."
|
|
339
340
|
)
|
|
340
341
|
# not running can be Idle or Finished
|
|
341
|
-
if
|
|
342
|
+
if status.get("failed"):
|
|
342
343
|
app.echo_error(message, nl=True, err=False)
|
|
343
|
-
elif
|
|
344
|
+
elif status.get("cancelled"):
|
|
344
345
|
app.echo_warning(message)
|
|
345
346
|
elif status["statusName"] == "Idle":
|
|
346
347
|
app.echo_info(message)
|
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
"""The main command line interface."""
|
|
2
|
+
|
|
2
3
|
import ast
|
|
3
4
|
import configparser
|
|
4
5
|
import json
|
|
5
6
|
import os
|
|
6
7
|
import re
|
|
7
8
|
import subprocess # nosec
|
|
8
|
-
from datetime import
|
|
9
|
+
from datetime import datetime, timezone
|
|
9
10
|
from os import environ as env
|
|
10
11
|
from os import getenv
|
|
11
12
|
from pathlib import Path
|
|
@@ -21,6 +22,7 @@ from pygments.lexers import get_lexer_by_name
|
|
|
21
22
|
from rich import box
|
|
22
23
|
from rich.console import Console
|
|
23
24
|
from rich.table import Table
|
|
25
|
+
from urllib3.exceptions import InsecureRequestWarning
|
|
24
26
|
|
|
25
27
|
from cmem_cmemc.exceptions import InvalidConfigurationError
|
|
26
28
|
|
|
@@ -47,6 +49,8 @@ KNOWN_CONFIG_KEYS = {
|
|
|
47
49
|
|
|
48
50
|
KNOWN_SECRET_KEYS = ("OAUTH_PASSWORD", "OAUTH_CLIENT_SECRET", "OAUTH_ACCESS_TOKEN")
|
|
49
51
|
|
|
52
|
+
SSL_VERIFY_WARNING = "SSL verification is disabled (SSL_VERIFY=False)."
|
|
53
|
+
|
|
50
54
|
|
|
51
55
|
class ApplicationContext:
|
|
52
56
|
"""Context of the command line interface."""
|
|
@@ -90,7 +94,7 @@ class ApplicationContext:
|
|
|
90
94
|
def get_template_data(self) -> dict[str, str]:
|
|
91
95
|
"""Get the template data dict with vars from the context."""
|
|
92
96
|
data: dict[str, str] = {}
|
|
93
|
-
today = str(datetime.now(tz=
|
|
97
|
+
today = str(datetime.now(tz=timezone.utc).date())
|
|
94
98
|
data.update(date=today)
|
|
95
99
|
if self.connection is not None:
|
|
96
100
|
data.update(connection=self.connection.name)
|
|
@@ -142,12 +146,6 @@ class ApplicationContext:
|
|
|
142
146
|
self.set_credential_from_process(_, _ + "_PROCESS", config)
|
|
143
147
|
|
|
144
148
|
self.echo_debug(f"CA bundle loaded from {cmempy_config.get_requests_ca_bundle()}")
|
|
145
|
-
# If cert validation is disabled, output a warning
|
|
146
|
-
# Also disable library warnings:
|
|
147
|
-
# https://urllib3.readthedocs.io/en/latest/advanced-usage.html
|
|
148
|
-
if not cmempy_config.get_ssl_verify():
|
|
149
|
-
self.echo_warning("SSL verification is disabled (SSL_VERIFY=False).")
|
|
150
|
-
urllib3.disable_warnings()
|
|
151
149
|
return
|
|
152
150
|
|
|
153
151
|
def set_connection_from_params(self, params: dict) -> None:
|
|
@@ -192,7 +190,7 @@ class ApplicationContext:
|
|
|
192
190
|
self.config = self.get_config()
|
|
193
191
|
self.connection = None
|
|
194
192
|
if section_string is None or section_string == "":
|
|
195
|
-
self.echo_debug("No config given, use API
|
|
193
|
+
self.echo_debug("No config given, use API defaults or environment connection.")
|
|
196
194
|
elif section_string not in self.config:
|
|
197
195
|
raise InvalidConfigurationError(
|
|
198
196
|
f"There is no connection '{section_string}' configured in "
|
|
@@ -202,6 +200,14 @@ class ApplicationContext:
|
|
|
202
200
|
self.echo_debug(f"Use connection config: {section_string}")
|
|
203
201
|
self.connection = self.config[section_string]
|
|
204
202
|
self.configure_cmempy(self.connection)
|
|
203
|
+
|
|
204
|
+
# If cert validation is disabled, output a warning
|
|
205
|
+
# Also disable library warnings:
|
|
206
|
+
# https://urllib3.readthedocs.io/en/latest/advanced-usage.html#tls-warnings
|
|
207
|
+
if not cmempy_config.get_ssl_verify():
|
|
208
|
+
self.echo_warning(SSL_VERIFY_WARNING)
|
|
209
|
+
urllib3.disable_warnings(category=InsecureRequestWarning)
|
|
210
|
+
|
|
205
211
|
return self.connection
|
|
206
212
|
|
|
207
213
|
def get_config_file(self) -> Path:
|
|
@@ -262,7 +268,7 @@ class ApplicationContext:
|
|
|
262
268
|
"""Output a debug message if --debug is enabled."""
|
|
263
269
|
# pylint: disable=invalid-name
|
|
264
270
|
if self.debug:
|
|
265
|
-
now = datetime.now(tz=
|
|
271
|
+
now = datetime.now(tz=timezone.utc)
|
|
266
272
|
click.secho(f"[{now!s}] {message}", err=True, dim=True)
|
|
267
273
|
|
|
268
274
|
def echo_info(self, message: str | list[str] | set[str], nl: bool = True, fg: str = "") -> None:
|
|
@@ -1,10 +1,12 @@
|
|
|
1
1
|
"""Utility functions for CLI interface."""
|
|
2
|
+
|
|
2
3
|
import json
|
|
3
4
|
import os
|
|
4
5
|
import re
|
|
6
|
+
import sys
|
|
5
7
|
import unicodedata
|
|
6
8
|
from dataclasses import dataclass
|
|
7
|
-
from importlib.metadata import version
|
|
9
|
+
from importlib.metadata import version as cmemc_version
|
|
8
10
|
from pathlib import Path
|
|
9
11
|
|
|
10
12
|
import requests
|
|
@@ -21,7 +23,22 @@ from cmem_cmemc.context import ApplicationContext
|
|
|
21
23
|
|
|
22
24
|
def get_version() -> str:
|
|
23
25
|
"""Get the current version or SNAPSHOT."""
|
|
24
|
-
return
|
|
26
|
+
return cmemc_version("cmem-cmemc")
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
def check_python_version(ctx: ApplicationContext) -> None:
|
|
30
|
+
"""Check the runtime python version and warn or error."""
|
|
31
|
+
version = sys.version_info
|
|
32
|
+
major_expected = [3]
|
|
33
|
+
minor_expected = [10, 11, 12]
|
|
34
|
+
if version.major not in major_expected:
|
|
35
|
+
ctx.echo_error(f"Error: cmemc can not be executed with Python {version.major}.")
|
|
36
|
+
sys.exit(1)
|
|
37
|
+
if version.minor not in minor_expected and not ctx.is_completing():
|
|
38
|
+
ctx.echo_warning(
|
|
39
|
+
"Warning: You are running cmemc under a non-tested python "
|
|
40
|
+
f"environment ({version.major}.{version.minor})."
|
|
41
|
+
)
|
|
25
42
|
|
|
26
43
|
|
|
27
44
|
def extract_error_message(error: Exception) -> str:
|
|
@@ -295,6 +312,8 @@ def get_published_packages() -> list[PublishedPackage]:
|
|
|
295
312
|
name = _.findChildren(class_="package-snippet__name")[0].getText()
|
|
296
313
|
if name == "cmem-plugin-base":
|
|
297
314
|
continue
|
|
315
|
+
if not name.startswith("cmem-plugin-"):
|
|
316
|
+
continue
|
|
298
317
|
description = _.findChildren(class_="package-snippet__description")[0].getText()
|
|
299
318
|
package_version = _.findChildren(class_="package-snippet__version")[0].getText()
|
|
300
319
|
published = _.findChildren(name="time")[0].attrs["datetime"]
|
|
@@ -1,9 +1,27 @@
|
|
|
1
1
|
[tool.poetry]
|
|
2
2
|
name = "cmem-cmemc"
|
|
3
|
-
version = "24.1.
|
|
3
|
+
version = "24.1.3"
|
|
4
4
|
description = "Command line client for eccenca Corporate Memory"
|
|
5
5
|
license = "Apache 2.0"
|
|
6
|
-
classifiers = [
|
|
6
|
+
classifiers = [
|
|
7
|
+
"Development Status :: 5 - Production/Stable",
|
|
8
|
+
"Environment :: Console",
|
|
9
|
+
"Intended Audience :: Customer Service",
|
|
10
|
+
"Intended Audience :: Developers",
|
|
11
|
+
"Intended Audience :: Information Technology",
|
|
12
|
+
"Intended Audience :: Science/Research",
|
|
13
|
+
"Intended Audience :: System Administrators",
|
|
14
|
+
"License :: OSI Approved :: Apache Software License",
|
|
15
|
+
"Natural Language :: English",
|
|
16
|
+
"Operating System :: OS Independent",
|
|
17
|
+
"Programming Language :: Python :: 3 :: Only",
|
|
18
|
+
"Programming Language :: Python :: 3.10",
|
|
19
|
+
"Programming Language :: Python :: 3.11",
|
|
20
|
+
"Programming Language :: Python :: 3.12",
|
|
21
|
+
"Topic :: Software Development :: Testing",
|
|
22
|
+
"Topic :: Database", "Topic :: Utilities",
|
|
23
|
+
"Topic :: Utilities"
|
|
24
|
+
]
|
|
7
25
|
homepage = "https://eccenca.com/go/cmemc"
|
|
8
26
|
authors = ["eccenca <cmempy-developer@eccenca.com>", "Sebastian Tramp <sebastian.tramp@eccenca.com>"]
|
|
9
27
|
readme = "README-public.md"
|
|
@@ -15,7 +33,7 @@ certifi = ">=2023.5.7"
|
|
|
15
33
|
click = "^8.1.7"
|
|
16
34
|
click-didyoumean = "^0.3.0"
|
|
17
35
|
click-help-colors = "^0.9.1"
|
|
18
|
-
cmem-cmempy = "==24.1.
|
|
36
|
+
cmem-cmempy = "==24.1.1"
|
|
19
37
|
# cmem-cmempy = { path = "cmempy", develop = true }
|
|
20
38
|
configparser = "^5.3.0"
|
|
21
39
|
jinja2 = "^3.1.2"
|
|
@@ -42,7 +60,7 @@ pytest = "^7.3.1"
|
|
|
42
60
|
pytest-cov = "^4.1.0"
|
|
43
61
|
pytest-dotenv = "^0.5.2"
|
|
44
62
|
pytest-memray = { version = "^1.5.0", markers = "platform_system != 'Windows'" }
|
|
45
|
-
ruff = "^0.
|
|
63
|
+
ruff = "^0.3.4"
|
|
46
64
|
safety = "^1.10.3"
|
|
47
65
|
types-requests = "^2.31.0.10"
|
|
48
66
|
types-six = "^1.16.21.9"
|
|
@@ -92,5 +110,6 @@ ignore = [
|
|
|
92
110
|
"PD", # opinionated linting for pandas code
|
|
93
111
|
"S101", # use of assert detected
|
|
94
112
|
"TRY003", # Avoid specifying long messages outside the exception class
|
|
113
|
+
"UP017", # We need compatibility 3.10 at the moment
|
|
95
114
|
]
|
|
96
115
|
|
cmem_cmemc-24.1.1rc1/PKG-INFO
DELETED
|
@@ -1,68 +0,0 @@
|
|
|
1
|
-
Metadata-Version: 2.1
|
|
2
|
-
Name: cmem-cmemc
|
|
3
|
-
Version: 24.1.1rc1
|
|
4
|
-
Summary: Command line client for eccenca Corporate Memory
|
|
5
|
-
Home-page: https://eccenca.com/go/cmemc
|
|
6
|
-
License: Apache 2.0
|
|
7
|
-
Author: eccenca
|
|
8
|
-
Author-email: cmempy-developer@eccenca.com
|
|
9
|
-
Requires-Python: >=3.9,<4.0
|
|
10
|
-
Classifier: Development Status :: 5 - Production/Stable
|
|
11
|
-
Classifier: Intended Audience :: Developers
|
|
12
|
-
Classifier: Intended Audience :: Science/Research
|
|
13
|
-
Classifier: Intended Audience :: System Administrators
|
|
14
|
-
Classifier: License :: OSI Approved :: Apache Software License
|
|
15
|
-
Classifier: License :: Other/Proprietary License
|
|
16
|
-
Classifier: Programming Language :: Python :: 3
|
|
17
|
-
Classifier: Programming Language :: Python :: 3.9
|
|
18
|
-
Classifier: Programming Language :: Python :: 3.10
|
|
19
|
-
Classifier: Programming Language :: Python :: 3.11
|
|
20
|
-
Classifier: Programming Language :: Python :: 3.12
|
|
21
|
-
Classifier: Topic :: Database
|
|
22
|
-
Classifier: Topic :: Software Development :: Testing
|
|
23
|
-
Classifier: Topic :: Utilities
|
|
24
|
-
Requires-Dist: beautifulsoup4 (>=4.12.2,<5.0.0)
|
|
25
|
-
Requires-Dist: certifi (>=2023.5.7)
|
|
26
|
-
Requires-Dist: click (>=8.1.7,<9.0.0)
|
|
27
|
-
Requires-Dist: click-didyoumean (>=0.3.0,<0.4.0)
|
|
28
|
-
Requires-Dist: click-help-colors (>=0.9.1,<0.10.0)
|
|
29
|
-
Requires-Dist: cmem-cmempy (==24.1.0)
|
|
30
|
-
Requires-Dist: configparser (>=5.3.0,<6.0.0)
|
|
31
|
-
Requires-Dist: jinja2 (>=3.1.2,<4.0.0)
|
|
32
|
-
Requires-Dist: natsort (>=8.3.1,<9.0.0)
|
|
33
|
-
Requires-Dist: prometheus-client (>=0.16.0,<0.17.0)
|
|
34
|
-
Requires-Dist: pygments (>=2.15.1,<3.0.0)
|
|
35
|
-
Requires-Dist: pyjwt (>=2.7.0,<3.0.0)
|
|
36
|
-
Requires-Dist: python-dateutil (>=2.8.2,<3.0.0)
|
|
37
|
-
Requires-Dist: requests (>=2.31.0,<3.0.0)
|
|
38
|
-
Requires-Dist: rich (>=13.7.0,<14.0.0)
|
|
39
|
-
Requires-Dist: six (>=1.16.0,<2.0.0)
|
|
40
|
-
Requires-Dist: timeago (>=1.0.16,<2.0.0)
|
|
41
|
-
Requires-Dist: treelib (>=1.6.4,<2.0.0)
|
|
42
|
-
Requires-Dist: urllib3 (>=2,<3)
|
|
43
|
-
Description-Content-Type: text/markdown
|
|
44
|
-
|
|
45
|
-
# cmemc
|
|
46
|
-
|
|
47
|
-
cmemc is the official command line client for [eccenca Corporate Memory](https://documentation.eccenca.com/).
|
|
48
|
-
|
|
49
|
-
## Installation
|
|
50
|
-
|
|
51
|
-
In order to install the cmemc, run:
|
|
52
|
-
|
|
53
|
-
pipx install cmem-cmemc
|
|
54
|
-
|
|
55
|
-
Of course you can install cmemc also with pip, but we recommend [pipx](https://pypa.github.io/pipx/) for normal desktop usage.
|
|
56
|
-
|
|
57
|
-
## Configuration and Usage
|
|
58
|
-
|
|
59
|
-
cmemc is intended for System Administrators and Linked Data Expert, who wants to automate and remote control activities on eccenca Corporate Memory.
|
|
60
|
-
|
|
61
|
-
The cmemc manual including basic usage pattern, configuration as well as a command reference is available at:
|
|
62
|
-
|
|
63
|
-
[https://eccenca.com/go/cmemc](https://eccenca.com/go/cmemc)
|
|
64
|
-
|
|
65
|
-
cmemc only works with Python 3 and refuses to work with Python 2.x.
|
|
66
|
-
In addition to that, cmemc will warn you in case an untested Python environment is used.
|
|
67
|
-
|
|
68
|
-
|
|
@@ -1,23 +0,0 @@
|
|
|
1
|
-
# cmemc
|
|
2
|
-
|
|
3
|
-
cmemc is the official command line client for [eccenca Corporate Memory](https://documentation.eccenca.com/).
|
|
4
|
-
|
|
5
|
-
## Installation
|
|
6
|
-
|
|
7
|
-
In order to install the cmemc, run:
|
|
8
|
-
|
|
9
|
-
pipx install cmem-cmemc
|
|
10
|
-
|
|
11
|
-
Of course you can install cmemc also with pip, but we recommend [pipx](https://pypa.github.io/pipx/) for normal desktop usage.
|
|
12
|
-
|
|
13
|
-
## Configuration and Usage
|
|
14
|
-
|
|
15
|
-
cmemc is intended for System Administrators and Linked Data Expert, who wants to automate and remote control activities on eccenca Corporate Memory.
|
|
16
|
-
|
|
17
|
-
The cmemc manual including basic usage pattern, configuration as well as a command reference is available at:
|
|
18
|
-
|
|
19
|
-
[https://eccenca.com/go/cmemc](https://eccenca.com/go/cmemc)
|
|
20
|
-
|
|
21
|
-
cmemc only works with Python 3 and refuses to work with Python 2.x.
|
|
22
|
-
In addition to that, cmemc will warn you in case an untested Python environment is used.
|
|
23
|
-
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|