From aa3d5465ecff6a4abf13bd7a79bf489a5bbd9e3a Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 31 Jan 2024 15:33:35 +0800 Subject: [PATCH 1/9] setup --- .github/workflows/codeql-analysis.yml | 75 +++++++++++++++++++++++++++ .github/workflows/publish-pypi.yml | 45 ++++++++++++++++ setup.py | 5 +- 3 files changed, 121 insertions(+), 4 deletions(-) create mode 100644 .github/workflows/codeql-analysis.yml create mode 100644 .github/workflows/publish-pypi.yml diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml new file mode 100644 index 000000000..0614a5fdf --- /dev/null +++ b/.github/workflows/codeql-analysis.yml @@ -0,0 +1,75 @@ +# For most projects, this workflow file will not need changing; you simply need +# to commit it to your repository. +# +# You may wish to alter this file to override the set of languages analyzed, +# or to provide custom queries or build logic. +# +# ******** NOTE ******** +# We have attempted to detect the languages in your repository. Please check +# the `language` matrix defined below to confirm you have the correct set of +# supported CodeQL languages. +# +name: "CodeQL" + +on: + push: + branches: [ master ] + pull_request: + # The branches below must be a subset of the branches above + branches: [ master ] + schedule: + - cron: '33 20 * * 5' + +jobs: + analyze: + name: Analyze + runs-on: ubuntu-latest + permissions: + actions: read + contents: read + security-events: write + + strategy: + fail-fast: false + matrix: + language: [ 'python' ] + # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] + # Learn more about CodeQL language support at https://git.io/codeql-language-support + + steps: + - name: Checkout repository + uses: actions/checkout@v2 + with: + lfs: true + + - name: Checkout LFS objects + run: git lfs pull + + # Initializes the CodeQL tools for scanning. + - name: Initialize CodeQL + uses: github/codeql-action/init@v1 + with: + languages: ${{ matrix.language }} + # If you wish to specify custom queries, you can do so here or in a config file. + # By default, queries listed here will override any specified in a config file. + # Prefix the list here with "+" to use these queries and those in the config file. + # queries: ./path/to/local/query, your-org/your-repo/queries@main + + # Autobuild attempts to build any compiled languages (C/C++, C#, or Java). + # If this step fails, then you should remove it and run the build manually (see below) + - name: Autobuild + uses: github/codeql-action/autobuild@v1 + + # ℹī¸ Command-line programs to run using the OS shell. + # 📚 https://git.io/JvXDl + + # ✏ī¸ If the Autobuild fails above, remove it and uncomment the following three lines + # and modify them (or add more) to build your code if your project + # uses a compiled language + + #- run: | + # make bootstrap + # make release + + - name: Perform CodeQL Analysis + uses: github/codeql-action/analyze@v1 diff --git a/.github/workflows/publish-pypi.yml b/.github/workflows/publish-pypi.yml new file mode 100644 index 000000000..24533eff5 --- /dev/null +++ b/.github/workflows/publish-pypi.yml @@ -0,0 +1,45 @@ +name: Publish Python 🐍 distributions đŸ“Ļ to PyPI and TestPyPI + +on: + push: + tags: + - '*' + workflow_dispatch: + + #TODO: Gate last step on release create? Unclear how to do the step's `if` + #release: + # types: [created] + +jobs: + build-n-publish: + name: Build and publish Python 🐍 distributions đŸ“Ļ to PyPI and TestPyPI + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + with: # fetch tag for versioneer + fetch-depth: 0 + + - name: Set up Python 3.7 + uses: actions/setup-python@v4 + with: + python-version: '3.7' + + - name: Install pypa/build + run: >- + python -m pip install -e .[build] + + - name: Build a binary wheel and a source tarball + run: >- + ./bin/build.sh + + - name: Publish distribution đŸ“Ļ to Test PyPI + uses: pypa/gh-action-pypi-publish@release/v1 + with: + password: ${{ secrets.PYPI_TEST }} + repository_url: https://test.pypi.org/legacy/ + + - name: Publish distribution đŸ“Ļ to PyPI + uses: pypa/gh-action-pypi-publish@release/v1 + with: + password: ${{ secrets.CUCAT_PYPI }} diff --git a/setup.py b/setup.py index 2b7bcbf52..eb3fba762 100644 --- a/setup.py +++ b/setup.py @@ -10,13 +10,10 @@ def unique_flatten_dict(d): 'numpy', 'pandas', 'setuptools', - 'logging', - 'typing', + 'typing-extensions', 'pyarrow>=0.15.0', 'scikit-learn==1.3.2', 'flake8>=5.0', - 'mypy', - 'pytest', 'psutil', 'build', 'dirty-cat' From 5f66695e2e5ab123ca9771470e023ac159ab3062 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 31 Jan 2024 15:44:26 +0800 Subject: [PATCH 2/9] setup lint --- setup.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/setup.py b/setup.py index eb3fba762..cfda044c6 100644 --- a/setup.py +++ b/setup.py @@ -16,7 +16,8 @@ def unique_flatten_dict(d): 'flake8>=5.0', 'psutil', 'build', - 'dirty-cat' + 'dirty-cat', + 'sphinx_autodoc_typehints==1.11.1', # 'cuml', ## cannot test on github actions # 'cudf', # 'cupy' @@ -38,6 +39,9 @@ def unique_flatten_dict(d): **dev_extras, + #kitchen sink for contributors, skips ai + 'dev': unique_flatten_dict(dev_extras), + } setup( From c8c61f4f2fe72cc796e51288a8153912d1e45b5e Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 31 Jan 2024 15:57:25 +0800 Subject: [PATCH 3/9] setup linting --- docs/source/index.rst | 46 +++++++++++++++++++++++++++++++++++++++++++ setup.py | 1 - 2 files changed, 46 insertions(+), 1 deletion(-) create mode 100644 docs/source/index.rst diff --git a/docs/source/index.rst b/docs/source/index.rst new file mode 100644 index 000000000..a474ac86f --- /dev/null +++ b/docs/source/index.rst @@ -0,0 +1,46 @@ +Cu_Cat: end-to-end GPU Feature Encoding +======================================== +.. only:: html + + .. image:: https://readthedocs.org/projects/cu_cat/badge/?version=latest + :target: https://cu_cat.readthedocs.io/en/latest/?badge=latest + :alt: Documentation Status + + + .. image:: https://github.com/graphistry/cu-cat/workflows/CI%20Tests/badge.svg + :target: https://github.com/graphistry/cu-cat/actions?query=workflow%3ACI + :alt: Build Status + + + .. image:: https://github.com/graphistry/cu-cat/workflows/CodeQL/badge.svg + :target: https://github.com/graphistry/cu-cat/actions?query=workflow%3ACodeQL + :alt: CodeQL Status + + .. image:: https://img.shields.io/pypi/v/graphistry.svg + :target: https://pypi.python.org/pypi/cu_cat + :alt: PyPi Status + +`cu-cat` is an end-to-end gpu Python library that encodes categorical variables into machine-learnable numerics. +It is a cuda accelerated port of what was dirty_cat, now rebranded as skrub, and allows more ambitious interactive analysis & real-time pipelines! + +.. toctree:: + :maxdepth: 3 + + cu_cat + modules + +Articles +================== +* `Cu_Cat: Hello interactive demo `_ +* `Cu_Cat + Single Cell `_ +* `Cu_Cat + Chemical Mapping `_ +* `Cu_Cat + Metagenomics `_ + + +Indices and tables +================== + +* :ref:`genindex` +* :ref:`modindex` +* :ref:`search` + diff --git a/setup.py b/setup.py index cfda044c6..2b9b64149 100644 --- a/setup.py +++ b/setup.py @@ -17,7 +17,6 @@ def unique_flatten_dict(d): 'psutil', 'build', 'dirty-cat', - 'sphinx_autodoc_typehints==1.11.1', # 'cuml', ## cannot test on github actions # 'cudf', # 'cupy' From 61853669a1655e5121f1ec9b0c33bcc69cce05e9 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 31 Jan 2024 16:13:27 +0800 Subject: [PATCH 4/9] more setup --- docs/source/cu_cat.rst | 31 +++++++++++++++++++++++++++++++ docs/source/index.rst | 12 ++---------- 2 files changed, 33 insertions(+), 10 deletions(-) create mode 100644 docs/source/cu_cat.rst diff --git a/docs/source/cu_cat.rst b/docs/source/cu_cat.rst new file mode 100644 index 000000000..2a3f2ffd9 --- /dev/null +++ b/docs/source/cu_cat.rst @@ -0,0 +1,31 @@ +Dependency_Manager +================== + +.. automodule:: cu_cat._dep_manager + :members: + :undoc-members: + :show-inheritance: + +Gap_Encoder +================== + +.. automodule:: cu_cat._gap_encoder + :members: + :undoc-members: + :show-inheritance: + +Table_Vectorizer +================== + +.. automodule:: cu_cat._table_vectorizer + :members: + :undoc-members: + :show-inheritance: + +Versioneer +================== + +.. automodule:: graphistry._version + :members: + :undoc-members: + :show-inheritance: diff --git a/docs/source/index.rst b/docs/source/index.rst index a474ac86f..51320db66 100644 --- a/docs/source/index.rst +++ b/docs/source/index.rst @@ -3,7 +3,7 @@ Cu_Cat: end-to-end GPU Feature Encoding .. only:: html .. image:: https://readthedocs.org/projects/cu_cat/badge/?version=latest - :target: https://cu_cat.readthedocs.io/en/latest/?badge=latest + :target: https://cu-cat.readthedocs.io/en/latest/?badge=latest :alt: Documentation Status @@ -16,7 +16,7 @@ Cu_Cat: end-to-end GPU Feature Encoding :target: https://github.com/graphistry/cu-cat/actions?query=workflow%3ACodeQL :alt: CodeQL Status - .. image:: https://img.shields.io/pypi/v/graphistry.svg + .. image:: https://img.shields.io/pypi/v/cu_cat.svg :target: https://pypi.python.org/pypi/cu_cat :alt: PyPi Status @@ -36,11 +36,3 @@ Articles * `Cu_Cat + Chemical Mapping `_ * `Cu_Cat + Metagenomics `_ - -Indices and tables -================== - -* :ref:`genindex` -* :ref:`modindex` -* :ref:`search` - From 89c9989114b63e58892add0b81414880da0ef6b2 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 31 Jan 2024 16:43:20 +0800 Subject: [PATCH 5/9] versioneer back --- MANIFEST.in | 2 + cu_cat/VERSION.txt | 1 - cu_cat/_version.py | 526 +++++++++++++++++++++++++++++++++++++++++++++ setup.py | 12 +- 4 files changed, 534 insertions(+), 7 deletions(-) create mode 100644 MANIFEST.in delete mode 100644 cu_cat/VERSION.txt create mode 100644 cu_cat/_version.py diff --git a/MANIFEST.in b/MANIFEST.in new file mode 100644 index 000000000..c53e2453a --- /dev/null +++ b/MANIFEST.in @@ -0,0 +1,2 @@ +include versioneer.py +include cu_cat/_version.py diff --git a/cu_cat/VERSION.txt b/cu_cat/VERSION.txt deleted file mode 100644 index c85ddec14..000000000 --- a/cu_cat/VERSION.txt +++ /dev/null @@ -1 +0,0 @@ -0.09.09 diff --git a/cu_cat/_version.py b/cu_cat/_version.py new file mode 100644 index 000000000..b17a7b1f3 --- /dev/null +++ b/cu_cat/_version.py @@ -0,0 +1,526 @@ + +# This file helps to compute a version number in source trees obtained from +# git-archive tarball (such as those provided by githubs download-from-tag +# feature). Distribution tarballs (built by setup.py sdist) and build +# directories (produced by setup.py build) will contain a much shorter file +# that just contains the computed version number. + +# This file is released into the public domain. Generated by +# versioneer-0.19 (https://github.com/python-versioneer/python-versioneer) + +"""Git implementation of _version.py.""" + +import errno +import os +import re +import subprocess +import sys +from typing import Any + + +def get_keywords(): + """Get the keywords needed to look up the version information.""" + # these strings will be replaced by git during git-archive. + # setup.py/versioneer.py will grep for the variable names, so they must + # each be defined on a line of their own. _version.py will just call + # get_keywords(). + git_refnames = "$Format:%d$" + git_full = "$Format:%H$" + git_date = "$Format:%ci$" + keywords = {"refnames": git_refnames, "full": git_full, "date": git_date} + return keywords + + +class VersioneerConfig: + """Container for Versioneer configuration parameters.""" + + +def get_config(): + """Create, populate and return the VersioneerConfig() object.""" + # these strings are filled in when 'setup.py versioneer' creates + # _version.py + cfg = VersioneerConfig() + cfg.VCS = "git" + cfg.style = "pep440" + cfg.tag_prefix = "" + cfg.parentdir_prefix = "graphistry-" + cfg.versionfile_source = "graphistry/_version.py" + cfg.verbose = False + return cfg + + +class NotThisMethod(Exception): + """Exception raised if a method is not valid for the current scenario.""" + + +LONG_VERSION_PY : Any = {} +HANDLERS = {} + + +def register_vcs_handler(vcs, method): # decorator + """Create decorator to mark a method as the handler of a VCS.""" + def decorate(f): + """Store f in HANDLERS[vcs][method].""" + if vcs not in HANDLERS: + HANDLERS[vcs] = {} + HANDLERS[vcs][method] = f + return f + return decorate + + +def run_command(commands, args, cwd=None, verbose=False, hide_stderr=False, + env=None): + """Call the given command(s).""" + assert isinstance(commands, list) + p = None + for c in commands: + try: + dispcmd = str([c] + args) + # remember shell=False, so use git.cmd on windows, not just git + p = subprocess.Popen([c] + args, cwd=cwd, env=env, + stdout=subprocess.PIPE, + stderr=(subprocess.PIPE if hide_stderr + else None)) + break + except EnvironmentError: + e = sys.exc_info()[1] + if e.errno == errno.ENOENT: + continue + if verbose: + print("unable to run %s" % dispcmd) + print(e) + return None, None + else: + if verbose: + print("unable to find command, tried %s" % (commands,)) + return None, None + stdout = p.communicate()[0].strip().decode() + if p.returncode != 0: + if verbose: + print("unable to run %s (error)" % dispcmd) + print("stdout was %s" % stdout) + return None, p.returncode + return stdout, p.returncode + + +def versions_from_parentdir(parentdir_prefix, root, verbose): + """Try to determine the version from the parent directory name. + + Source tarballs conventionally unpack into a directory that includes both + the project name and a version string. We will also support searching up + two directory levels for an appropriately named parent directory + """ + rootdirs = [] + + for i in range(3): + dirname = os.path.basename(root) + if dirname.startswith(parentdir_prefix): + return {"version": dirname[len(parentdir_prefix):], + "full-revisionid": None, + "dirty": False, "error": None, "date": None} + else: + rootdirs.append(root) + root = os.path.dirname(root) # up a level + + if verbose: + print("Tried directories %s but none started with prefix %s" % + (str(rootdirs), parentdir_prefix)) + raise NotThisMethod("rootdir doesn't start with parentdir_prefix") + + +@register_vcs_handler("git", "get_keywords") +def git_get_keywords(versionfile_abs): + """Extract version information from the given file.""" + # the code embedded in _version.py can just fetch the value of these + # keywords. When used from setup.py, we don't want to import _version.py, + # so we do it with a regexp instead. This function is not used from + # _version.py. + keywords = {} + try: + f = open(versionfile_abs, "r") + for line in f.readlines(): + if line.strip().startswith("git_refnames ="): + mo = re.search(r'=\s*"(.*)"', line) + if mo: + keywords["refnames"] = mo.group(1) + if line.strip().startswith("git_full ="): + mo = re.search(r'=\s*"(.*)"', line) + if mo: + keywords["full"] = mo.group(1) + if line.strip().startswith("git_date ="): + mo = re.search(r'=\s*"(.*)"', line) + if mo: + keywords["date"] = mo.group(1) + f.close() + except EnvironmentError: + pass + return keywords + + +@register_vcs_handler("git", "keywords") +def git_versions_from_keywords(keywords, tag_prefix, verbose): + """Get version information from git keywords.""" + if not keywords: + raise NotThisMethod("no keywords at all, weird") + date = keywords.get("date") + if date is not None: + # Use only the last line. Previous lines may contain GPG signature + # information. + date = date.splitlines()[-1] + + # git-2.2.0 added "%cI", which expands to an ISO-8601 -compliant + # datestamp. However we prefer "%ci" (which expands to an "ISO-8601 + # -like" string, which we must then edit to make compliant), because + # it's been around since git-1.5.3, and it's too difficult to + # discover which version we're using, or to work around using an + # older one. + date = date.strip().replace(" ", "T", 1).replace(" ", "", 1) + refnames = keywords["refnames"].strip() + if refnames.startswith("$Format"): + if verbose: + print("keywords are unexpanded, not using") + raise NotThisMethod("unexpanded keywords, not a git-archive tarball") + refs = set([r.strip() for r in refnames.strip("()").split(",")]) + # starting in git-1.8.3, tags are listed as "tag: foo-1.0" instead of + # just "foo-1.0". If we see a "tag: " prefix, prefer those. + TAG = "tag: " + tags = set([r[len(TAG):] for r in refs if r.startswith(TAG)]) + if not tags: + # Either we're using git < 1.8.3, or there really are no tags. We use + # a heuristic: assume all version tags have a digit. The old git %d + # expansion behaves like git log --decorate=short and strips out the + # refs/heads/ and refs/tags/ prefixes that would let us distinguish + # between branches and tags. By ignoring refnames without digits, we + # filter out many common branch names like "release" and + # "stabilization", as well as "HEAD" and "master". + tags = set([r for r in refs if re.search(r'\d', r)]) + if verbose: + print("discarding '%s', no digits" % ",".join(refs - tags)) + if verbose: + print("likely tags: %s" % ",".join(sorted(tags))) + for ref in sorted(tags): + # sorting will prefer e.g. "2.0" over "2.0rc1" + if ref.startswith(tag_prefix): + r = ref[len(tag_prefix):] + if verbose: + print("picking %s" % r) + return {"version": r, + "full-revisionid": keywords["full"].strip(), + "dirty": False, "error": None, + "date": date} + # no suitable tags, so version is "0+unknown", but full hex is still there + if verbose: + print("no suitable tags, using unknown + full revision id") + return {"version": "0+unknown", + "full-revisionid": keywords["full"].strip(), + "dirty": False, "error": "no suitable tags", "date": None} + + +@register_vcs_handler("git", "pieces_from_vcs") +def git_pieces_from_vcs(tag_prefix, root, verbose, run_command=run_command): + """Get version from 'git describe' in the root of the source tree. + + This only gets called if the git-archive 'subst' keywords were *not* + expanded, and _version.py hasn't already been rewritten with a short + version string, meaning we're inside a checked out source tree. + """ + GITS = ["git"] + if sys.platform == "win32": + GITS = ["git.cmd", "git.exe"] + + out, rc = run_command(GITS, ["rev-parse", "--git-dir"], cwd=root, + hide_stderr=True) + if rc != 0: + if verbose: + print("Directory %s not under git control" % root) + raise NotThisMethod("'git rev-parse --git-dir' returned error") + + # if there is a tag matching tag_prefix, this yields TAG-NUM-gHEX[-dirty] + # if there isn't one, this yields HEX[-dirty] (no NUM) + describe_out, rc = run_command(GITS, ["describe", "--tags", "--dirty", + "--always", "--long", + "--match", "%s*" % tag_prefix], + cwd=root) + # --long was added in git-1.5.5 + if describe_out is None: + raise NotThisMethod("'git describe' failed") + describe_out = describe_out.strip() + full_out, rc = run_command(GITS, ["rev-parse", "HEAD"], cwd=root) + if full_out is None: + raise NotThisMethod("'git rev-parse' failed") + full_out = full_out.strip() + + pieces = {} + pieces["long"] = full_out + pieces["short"] = full_out[:7] # maybe improved later + pieces["error"] = None + + # parse describe_out. It will be like TAG-NUM-gHEX[-dirty] or HEX[-dirty] + # TAG might have hyphens. + git_describe = describe_out + + # look for -dirty suffix + dirty = git_describe.endswith("-dirty") + pieces["dirty"] = dirty + if dirty: + git_describe = git_describe[:git_describe.rindex("-dirty")] + + # now we have TAG-NUM-gHEX or HEX + + if "-" in git_describe: + # TAG-NUM-gHEX + mo = re.search(r'^(.+)-(\d+)-g([0-9a-f]+)$', git_describe) + if not mo: + # unparseable. Maybe git-describe is misbehaving? + pieces["error"] = ("unable to parse git-describe output: '%s'" + % describe_out) + return pieces + + # tag + full_tag = mo.group(1) + if not full_tag.startswith(tag_prefix): + if verbose: + fmt = "tag '%s' doesn't start with prefix '%s'" + print(fmt % (full_tag, tag_prefix)) + pieces["error"] = ("tag '%s' doesn't start with prefix '%s'" + % (full_tag, tag_prefix)) + return pieces + pieces["closest-tag"] = full_tag[len(tag_prefix):] + + # distance: number of commits since tag + pieces["distance"] = int(mo.group(2)) + + # commit: short hex revision ID + pieces["short"] = mo.group(3) + + else: + # HEX: no tags + pieces["closest-tag"] = None + count_out, rc = run_command(GITS, ["rev-list", "HEAD", "--count"], + cwd=root) + pieces["distance"] = int(count_out) # total number of commits + + # commit date: see ISO-8601 comment in git_versions_from_keywords() + date = run_command(GITS, ["show", "-s", "--format=%ci", "HEAD"], + cwd=root)[0].strip() + # Use only the last line. Previous lines may contain GPG signature + # information. + date = date.splitlines()[-1] + pieces["date"] = date.strip().replace(" ", "T", 1).replace(" ", "", 1) + + return pieces + + +def plus_or_dot(pieces): + """Return a + if we don't already have one, else return a .""" + if "+" in pieces.get("closest-tag", ""): + return "." + return "+" + + +def render_pep440(pieces): + """Build up version string, with post-release "local version identifier". + + Our goal: TAG[+DISTANCE.gHEX[.dirty]] . Note that if you + get a tagged build and then dirty it, you'll get TAG+0.gHEX.dirty + + Exceptions: + 1: no tags. git_describe was just HEX. 0+untagged.DISTANCE.gHEX[.dirty] + """ + if pieces["closest-tag"]: + rendered = pieces["closest-tag"] + if pieces["distance"] or pieces["dirty"]: + rendered += plus_or_dot(pieces) + rendered += "%d.g%s" % (pieces["distance"], pieces["short"]) + if pieces["dirty"]: + rendered += ".dirty" + else: + # exception #1 + rendered = "0+untagged.%d.g%s" % (pieces["distance"], + pieces["short"]) + if pieces["dirty"]: + rendered += ".dirty" + return rendered + + +def render_pep440_pre(pieces): + """TAG[.post0.devDISTANCE] -- No -dirty. + + Exceptions: + 1: no tags. 0.post0.devDISTANCE + """ + if pieces["closest-tag"]: + rendered = pieces["closest-tag"] + if pieces["distance"]: + rendered += ".post0.dev%d" % pieces["distance"] + else: + # exception #1 + rendered = "0.post0.dev%d" % pieces["distance"] + return rendered + + +def render_pep440_post(pieces): + """TAG[.postDISTANCE[.dev0]+gHEX] . + + The ".dev0" means dirty. Note that .dev0 sorts backwards + (a dirty tree will appear "older" than the corresponding clean one), + but you shouldn't be releasing software with -dirty anyways. + + Exceptions: + 1: no tags. 0.postDISTANCE[.dev0] + """ + if pieces["closest-tag"]: + rendered = pieces["closest-tag"] + if pieces["distance"] or pieces["dirty"]: + rendered += ".post%d" % pieces["distance"] + if pieces["dirty"]: + rendered += ".dev0" + rendered += plus_or_dot(pieces) + rendered += "g%s" % pieces["short"] + else: + # exception #1 + rendered = "0.post%d" % pieces["distance"] + if pieces["dirty"]: + rendered += ".dev0" + rendered += "+g%s" % pieces["short"] + return rendered + + +def render_pep440_old(pieces): + """TAG[.postDISTANCE[.dev0]] . + + The ".dev0" means dirty. + + Exceptions: + 1: no tags. 0.postDISTANCE[.dev0] + """ + if pieces["closest-tag"]: + rendered = pieces["closest-tag"] + if pieces["distance"] or pieces["dirty"]: + rendered += ".post%d" % pieces["distance"] + if pieces["dirty"]: + rendered += ".dev0" + else: + # exception #1 + rendered = "0.post%d" % pieces["distance"] + if pieces["dirty"]: + rendered += ".dev0" + return rendered + + +def render_git_describe(pieces): + """TAG[-DISTANCE-gHEX][-dirty]. + + Like 'git describe --tags --dirty --always'. + + Exceptions: + 1: no tags. HEX[-dirty] (note: no 'g' prefix) + """ + if pieces["closest-tag"]: + rendered = pieces["closest-tag"] + if pieces["distance"]: + rendered += "-%d-g%s" % (pieces["distance"], pieces["short"]) + else: + # exception #1 + rendered = pieces["short"] + if pieces["dirty"]: + rendered += "-dirty" + return rendered + + +def render_git_describe_long(pieces): + """TAG-DISTANCE-gHEX[-dirty]. + + Like 'git describe --tags --dirty --always -long'. + The distance/hash is unconditional. + + Exceptions: + 1: no tags. HEX[-dirty] (note: no 'g' prefix) + """ + if pieces["closest-tag"]: + rendered = pieces["closest-tag"] + rendered += "-%d-g%s" % (pieces["distance"], pieces["short"]) + else: + # exception #1 + rendered = pieces["short"] + if pieces["dirty"]: + rendered += "-dirty" + return rendered + + +def render(pieces, style): + """Render the given version pieces into the requested style.""" + if pieces["error"]: + return {"version": "unknown", + "full-revisionid": pieces.get("long"), + "dirty": None, + "error": pieces["error"], + "date": None} + + if not style or style == "default": + style = "pep440" # the default + + if style == "pep440": + rendered = render_pep440(pieces) + elif style == "pep440-pre": + rendered = render_pep440_pre(pieces) + elif style == "pep440-post": + rendered = render_pep440_post(pieces) + elif style == "pep440-old": + rendered = render_pep440_old(pieces) + elif style == "git-describe": + rendered = render_git_describe(pieces) + elif style == "git-describe-long": + rendered = render_git_describe_long(pieces) + else: + raise ValueError("unknown style '%s'" % style) + + return {"version": rendered, "full-revisionid": pieces["long"], + "dirty": pieces["dirty"], "error": None, + "date": pieces.get("date")} + + +def get_versions(): + """Get version information or return default if unable to do so.""" + # I am in _version.py, which lives at ROOT/VERSIONFILE_SOURCE. If we have + # __file__, we can work backwards from there to the root. Some + # py2exe/bbfreeze/non-CPython implementations don't do __file__, in which + # case we can only use expanded keywords. + + cfg = get_config() + verbose = cfg.verbose + + try: + return git_versions_from_keywords(get_keywords(), cfg.tag_prefix, + verbose) + except NotThisMethod: + pass + + try: + root = os.path.realpath(__file__) + # versionfile_source is the relative path from the top of the source + # tree (where the .git directory might live) to this file. Invert + # this to find the root from __file__. + for i in cfg.versionfile_source.split('/'): + root = os.path.dirname(root) + except NameError: + return {"version": "0+unknown", "full-revisionid": None, + "dirty": None, + "error": "unable to find root of source tree", + "date": None} + + try: + pieces = git_pieces_from_vcs(cfg.tag_prefix, root, verbose) + return render(pieces, cfg.style) + except NotThisMethod: + pass + + try: + if cfg.parentdir_prefix: + return versions_from_parentdir(cfg.parentdir_prefix, root, verbose) + except NotThisMethod: + pass + + return {"version": "0+unknown", "full-revisionid": None, + "dirty": None, + "error": "unable to compute version", "date": None} diff --git a/setup.py b/setup.py index 2b9b64149..8f2aa40b6 100644 --- a/setup.py +++ b/setup.py @@ -1,7 +1,7 @@ #!/usr/bin/env python from setuptools import setup, find_packages -# # # # import versioneer +import versioneer # # #github def unique_flatten_dict(d): return list(set(sum( d.values(), [] ))) @@ -42,11 +42,11 @@ def unique_flatten_dict(d): 'dev': unique_flatten_dict(dev_extras), } - +# if __name__ == "__main__": setup( name='cu-cat', - version='v0.09.09', - # cmdclass=versioneer.get_cmdclass(), + version=versioneer.get_version(), + cmdclass=versioneer.get_cmdclass(), packages = find_packages(), platforms='any', description = 'An end-to-end gpu Python library that encodes categorical variables into machine-learnable numerics', @@ -54,13 +54,13 @@ def unique_flatten_dict(d): long_description_content_type='text/markdown', url='https://github.com/graphistry/cu-cat', download_url= 'https://github.com/graphistry/cu-cat', - python_requires='>=3.7', + python_requires='>3.7', author='The Graphistry Team', author_email='pygraphistry@graphistry.com', install_requires=core_requires, extras_require=extras_require, license='BSD', - + # dependency_links=['https://pypi.nvidia.com'], keywords=['cudf', 'cuml', 'GPU', 'Rapids'] ) From e9150c4402cc98645d76a659faa94767d19a73ba Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 31 Jan 2024 16:53:09 +0800 Subject: [PATCH 6/9] manifest add --- docs/source/index.rst | 7 +++++++ docs/source/modules.rst | 9 +++++++++ docs/source/versioneer.rst | 6 ++++++ 3 files changed, 22 insertions(+) create mode 100644 docs/source/modules.rst create mode 100644 docs/source/versioneer.rst diff --git a/docs/source/index.rst b/docs/source/index.rst index 51320db66..5b173a0fd 100644 --- a/docs/source/index.rst +++ b/docs/source/index.rst @@ -36,3 +36,10 @@ Articles * `Cu_Cat + Chemical Mapping `_ * `Cu_Cat + Metagenomics `_ +Indices and tables +================== + +* :ref:`genindex` +* :ref:`modindex` +* :ref:`search` + diff --git a/docs/source/modules.rst b/docs/source/modules.rst new file mode 100644 index 000000000..ced1d0941 --- /dev/null +++ b/docs/source/modules.rst @@ -0,0 +1,9 @@ +Modules +##################### + +.. toctree:: + :maxdepth: 4 + :caption: Contents: + + versioneer + diff --git a/docs/source/versioneer.rst b/docs/source/versioneer.rst new file mode 100644 index 000000000..1f5d4bae4 --- /dev/null +++ b/docs/source/versioneer.rst @@ -0,0 +1,6 @@ +.. versioneer module +.. ================= +.. toctree:: + :maxdepth: 2 + + graphistry.plugins_types From aab48b18c785c5a6ed5c7f287ffbd2547bf1235e Mon Sep 17 00:00:00 2001 From: Daniel Date: Fri, 2 Feb 2024 09:30:33 +0800 Subject: [PATCH 7/9] add pip versioneer --- setup.py | 1 + 1 file changed, 1 insertion(+) diff --git a/setup.py b/setup.py index 8f2aa40b6..38b21ee1a 100644 --- a/setup.py +++ b/setup.py @@ -16,6 +16,7 @@ def unique_flatten_dict(d): 'flake8>=5.0', 'psutil', 'build', + 'versioneer', 'dirty-cat', # 'cuml', ## cannot test on github actions # 'cudf', From 974f9c5ff79ac1ff2fa990cf77562b1ff146e31b Mon Sep 17 00:00:00 2001 From: Daniel Date: Fri, 2 Feb 2024 10:24:16 +0800 Subject: [PATCH 8/9] versioneer fix --- README.md | 31 +++++++++++++++++++++++++++++-- cu_cat/__init__.py | 5 +++-- cu_cat/_version.py | 4 ++-- docs/source/conf.py | 2 +- docs/source/cu_cat.rst | 2 +- docs/source/versioneer.rst | 2 +- setup.py | 7 +++---- 7 files changed, 40 insertions(+), 13 deletions(-) diff --git a/README.md b/README.md index 24b1f7a1e..52fc9248a 100644 --- a/README.md +++ b/README.md @@ -42,8 +42,35 @@ However, this graph does not mean to imply the trend goes on forever, as current GPU = colab T4 + 15gb mem and colab CPU + 12gb memory - -## Startup Code: +## Startup Code demonstrating speedup: + + ! pip install cu-cat dirty-cat + from time import time + from cu_cat._table_vectorizer import TableVectorizer as cu_TableVectorizer + from dirty_cat._table_vectorizer import TableVectorizer as dirty_TableVectorizer + from sklearn.datasets import fetch_20newsgroups + n_samples = 2000 # speed boost improves as n_samples increases, to the limit of gpu mem + + news, _ = fetch_20newsgroups( + shuffle=True, + random_state=1, + remove=("headers", "footers", "quotes"), + return_X_y=True, + ) + + news = news[:n_samples] + news=pd.DataFrame(news) + table_vec = cu_TableVectorizer() + t = time() + aa = table_vec.fit_transform((news)) + ct = time() - t + # if deps.dirty_cat: + t = time() + bb = dirty_TableVectorizer().fit_transform(news) + dt = time() - t + print(f"cu_cat: {ct:.2f}s, dirty_cat: {dt:.2f}s, speedup: {dt/ct:.2f}x") + >>> cu_cat: 58.76s, dirty_cat: 84.54s, speedup: 1.44x +## Enhanced Code using Graphistry: # !pip install graphistry[ai] ## future releases will have this by default !pip install git+https://github.com/graphistry/pygraphistry.git@dev/depman_gpufeat diff --git a/cu_cat/__init__.py b/cu_cat/__init__.py index cc48ad80c..0fe7a0c14 100644 --- a/cu_cat/__init__.py +++ b/cu_cat/__init__.py @@ -20,9 +20,10 @@ from ._gap_encoder import GapEncoder # type: ignore from ._table_vectorizer import SuperVectorizer, TableVectorizer -with open(_Path(__file__).parent / "VERSION.txt") as _fh: - __version__ = _fh.read().strip() +from ._version import get_versions +__version__ = get_versions()["version"] +del get_versions __all__ = [ "DatetimeEncoder", diff --git a/cu_cat/_version.py b/cu_cat/_version.py index b17a7b1f3..33f0c6a08 100644 --- a/cu_cat/_version.py +++ b/cu_cat/_version.py @@ -43,8 +43,8 @@ def get_config(): cfg.VCS = "git" cfg.style = "pep440" cfg.tag_prefix = "" - cfg.parentdir_prefix = "graphistry-" - cfg.versionfile_source = "graphistry/_version.py" + cfg.parentdir_prefix = "cu_cat-" + cfg.versionfile_source = "cu_cat/_version.py" cfg.verbose = False return cfg diff --git a/docs/source/conf.py b/docs/source/conf.py index 8a2c87380..7874ad331 100644 --- a/docs/source/conf.py +++ b/docs/source/conf.py @@ -21,7 +21,7 @@ # -- Project information ----------------------------------------------------- project = "CU_CAT" -copyright = "2023, Graphistry, Inc." +copyright = "2024, Graphistry, Inc." author = "Graphistry, Inc." # The full version, including alpha/beta/rc tags diff --git a/docs/source/cu_cat.rst b/docs/source/cu_cat.rst index 2a3f2ffd9..2bba5d6d4 100644 --- a/docs/source/cu_cat.rst +++ b/docs/source/cu_cat.rst @@ -25,7 +25,7 @@ Table_Vectorizer Versioneer ================== -.. automodule:: graphistry._version +.. automodule:: cu_cat._version :members: :undoc-members: :show-inheritance: diff --git a/docs/source/versioneer.rst b/docs/source/versioneer.rst index 1f5d4bae4..51dd74798 100644 --- a/docs/source/versioneer.rst +++ b/docs/source/versioneer.rst @@ -3,4 +3,4 @@ .. toctree:: :maxdepth: 2 - graphistry.plugins_types + cu_cat.plugins_types diff --git a/setup.py b/setup.py index 38b21ee1a..d2ea36112 100644 --- a/setup.py +++ b/setup.py @@ -2,7 +2,7 @@ from setuptools import setup, find_packages import versioneer -# # #github + def unique_flatten_dict(d): return list(set(sum( d.values(), [] ))) @@ -16,9 +16,8 @@ def unique_flatten_dict(d): 'flake8>=5.0', 'psutil', 'build', - 'versioneer', - 'dirty-cat', -# 'cuml', ## cannot test on github actions + 'dirty-cat', # only for pytest speed comparison +# 'cuml', # cannot test on github actions # 'cudf', # 'cupy' ] From 215271d2889073f57a66993946f55cbf98a9ec74 Mon Sep 17 00:00:00 2001 From: Daniel Date: Sat, 3 Feb 2024 21:25:32 +0800 Subject: [PATCH 9/9] remove conflicting files --- .git-blame-ignore-revs | 15 -------------- .readthedocs.yml | 28 -------------------------- README.md | 6 +----- cu_cat/tests/__init__.py | 0 examples/cu_cat.png | Bin 0 -> 79682 bytes pipupload.sh | 1 + pyproject.toml | 42 --------------------------------------- 7 files changed, 2 insertions(+), 90 deletions(-) delete mode 100644 .git-blame-ignore-revs delete mode 100644 .readthedocs.yml delete mode 100644 cu_cat/tests/__init__.py create mode 100644 examples/cu_cat.png create mode 100755 pipupload.sh delete mode 100644 pyproject.toml diff --git a/.git-blame-ignore-revs b/.git-blame-ignore-revs deleted file mode 100644 index 36a47fca2..000000000 --- a/.git-blame-ignore-revs +++ /dev/null @@ -1,15 +0,0 @@ -# Since git version 2.23, git-blame has a feature to ignore -# certain commits. -# -# This file contains a list of commits that are not likely what -# you are looking for in `git blame`. You can set this file as -# a default ignore file for blame by running the following -# command. -# -# $ git config blame.ignoreRevsFile .git-blame-ignore-revs - -# PR 314: MAINT Format the code-base to be pre-commit compliant -4c8e17882d602e4b314ce150ad345e59a88e2aab - -# PR 331: Make unnecessary API private -1bf617a079722e02cfb73f6a2d3dc88e8769a80f diff --git a/.readthedocs.yml b/.readthedocs.yml deleted file mode 100644 index 609e875f7..000000000 --- a/.readthedocs.yml +++ /dev/null @@ -1,28 +0,0 @@ -# .readthedocs.yml -# Read the Docs configuration file -# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details - -# Required -version: 2 - -# Build documentation in the docs/ directory with Sphinx -sphinx: - configuration: docs/source/conf.py - -build: - os: ubuntu-22.04 - tools: - python: "3.8" - -# Optionally build your docs in additional formats such as PDF -formats: - - pdf - - htmlzip - - epub - -python: - install: - - method: pip - path: . - extra_requirements: - - dev \ No newline at end of file diff --git a/README.md b/README.md index 52fc9248a..b8a8b12af 100644 --- a/README.md +++ b/README.md @@ -1,10 +1,6 @@ # **cu-cat** -``` - /\_/\ -( o.o ) - > ^ < -``` +drawing ****cu-cat**** is an end-to-end gpu Python library that encodes categorical variables into machine-learnable numerics. It is a cuda diff --git a/cu_cat/tests/__init__.py b/cu_cat/tests/__init__.py deleted file mode 100644 index e69de29bb..000000000 diff --git a/examples/cu_cat.png b/examples/cu_cat.png new file mode 100644 index 0000000000000000000000000000000000000000..bc0d524a85d23acbfaa523e777ef3620f99a2489 GIT binary patch literal 79682 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 O00003du literal 0 HcmV?d00001 diff --git a/pipupload.sh b/pipupload.sh new file mode 100755 index 000000000..9e00052da --- /dev/null +++ b/pipupload.sh @@ -0,0 +1 @@ +docker run -v "${PWD}/pypirc":/root/.pypirc -v "${PWD}":/repo -w /repo python bash -c "python3 setup.py bdist_wheel upload -r pypi" diff --git a/pyproject.toml b/pyproject.toml deleted file mode 100644 index 6e5815f2e..000000000 --- a/pyproject.toml +++ /dev/null @@ -1,42 +0,0 @@ -[build-system] -# Minimum requirements for the build system to execute. -requires = [ - "setuptools", - "wheel", -] -build-backend = "setuptools.build_meta" - -[tool.black] -line-length = 88 -target_version = ['py310', 'py311'] -preview = true -# Exclude irrelevant directories for formatting -exclude = ''' -/( - \.eggs - | \.git - | \.mypy_cache - | \.vscode - | \.pytest_cache - | \.idea - | build - | dist -)/ -''' - -[tool.isort] -profile = "black" - -[tool.pytest.ini_options] -filterwarnings = [ - # Turn deprecation warnings into errors - "error::FutureWarning", - "error::DeprecationWarning", - "error::numpy.VisibleDeprecationWarning", - - # Ignore warning from np.in1d since the future behavior is already the desired - # behavior. TODO remove when numpy min version >= 1.25. - 'ignore:elementwise\ comparison\ failed:FutureWarning', -] -addopts = "--doctest-modules" -# doctest_optionflags = "NORMALIZE_WHITESPACE ELLIPSIS"