mirror of
https://github.com/daylinmorgan/viv.git
synced 2024-11-09 19:13:14 -06:00
Compare commits
5 commits
852ce927c0
...
ea50661a04
Author | SHA1 | Date | |
---|---|---|---|
ea50661a04 | |||
3ab39c45c1 | |||
05d2146b97 | |||
bcad0e5696 | |||
ab1444ace7 |
4 changed files with 63 additions and 78 deletions
|
@ -129,12 +129,12 @@ python3 <(curl -fsSL viv.dayl.in/viv.py) run \
|
||||||
-s https://gist.githubusercontent.com/cs01/fa721a17a326e551ede048c5088f9e0f/raw/6bdfbb6e9c1132b1c38fdd2f195d4a24c540c324/pipx-demo.py
|
-s https://gist.githubusercontent.com/cs01/fa721a17a326e551ede048c5088f9e0f/raw/6bdfbb6e9c1132b1c38fdd2f195d4a24c540c324/pipx-demo.py
|
||||||
```
|
```
|
||||||
|
|
||||||
## Bonus: use `viv` with just standalone snippet (37LOC)
|
## Bonus: use `viv` with just standalone snippet (30LOC)
|
||||||
|
|
||||||
`--standalone` will auto-generate a mini-function version of `viv`
|
`--standalone` will auto-generate a mini-function version of `viv`
|
||||||
to accomplish the same basic task as using a local copy of `viv`.
|
to accomplish the same basic task as using a local copy of `viv`.
|
||||||
After generating this standalone shim you can freely use this script
|
After generating this standalone shim you can freely use this script
|
||||||
across Unix machines which have Python > 3.8.
|
across Unix machines which have Python > 3.8 and pip.
|
||||||
See [examples/black](https://github.com/daylinmorgan/viv/blob/dev/examples/black)
|
See [examples/black](https://github.com/daylinmorgan/viv/blob/dev/examples/black)
|
||||||
for output of the below command.
|
for output of the below command.
|
||||||
|
|
||||||
|
|
|
@ -1,47 +1,37 @@
|
||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
# AUTOGENERATED by viv (v2023.1001-dev)
|
# AUTOGENERATED by viv (v2024.1005-dev)
|
||||||
# see `python3 <(curl -fsSL viv.dayl.in/viv.py) --help`
|
# see `python3 <(curl -fsSL viv.dayl.in/viv.py) --help`
|
||||||
|
|
||||||
|
|
||||||
def _viv_use(*pkgs, track_exe=False, name=""):
|
def _viv_use(*pkgs, track_exe=False, name=""):
|
||||||
import hashlib, json, os, site, shutil, sys, venv # noqa
|
i, meta, add_meta = __import__, {}, lambda **kw: meta.update(**kw)
|
||||||
from pathlib import Path # noqa
|
P, ge, q, noop = i("pathlib").Path, i("os").getenv, i("sys").exit, lambda: None
|
||||||
from datetime import datetime # noqa
|
_if, p_str = lambda x, f: (noop, f)[x](), lambda x: f"{P(x).absolute().resolve()}"
|
||||||
from subprocess import run # noqa
|
|
||||||
|
|
||||||
if not {*map(type, pkgs)} == {str}:
|
|
||||||
raise ValueError(f"spec: {pkgs} is invalid")
|
|
||||||
|
|
||||||
meta = dict.fromkeys(("created", "accessed"), (t := str(datetime.today())))
|
|
||||||
runner = str(Path(__file__).absolute().resolve())
|
|
||||||
envvar = lambda x: os.getenv(f"VIV_{x}") # noqa
|
|
||||||
nopkgs = lambda p: not p.endswith(("dist-packages", "site-packages")) # noqa
|
nopkgs = lambda p: not p.endswith(("dist-packages", "site-packages")) # noqa
|
||||||
F, V, no_st = map(envvar, ("FORCE", "VERBOSE", "NO_SETUPTOOLS"))
|
_if({*map(type, spec := [*pkgs])} != {str}, lambda: q(f"spec: {pkgs} not string"))
|
||||||
base = Path(xdg) if (xdg := os.getenv("XDG_CACHE_HOME")) else Path.home() / ".cache"
|
F, V, NS = map(lambda x: ge(f"VIV_{x}"), ("FORCE", "VERBOSE", "NO_SETUPTOOLS"))
|
||||||
|
ST = not (NS and [x for x in spec if x.startswith("setuptools")])
|
||||||
|
add_meta(created=(now := str(i("datetime").datetime.today())))
|
||||||
|
base = P(xdg) if (xdg := ge("XDG_CACHE_HOME")) else P.home() / ".cache"
|
||||||
(cache := (base) / "viv/venvs").mkdir(parents=True, exist_ok=True)
|
(cache := (base) / "viv/venvs").mkdir(parents=True, exist_ok=True)
|
||||||
exe = str(Path(sys.executable).resolve()) if track_exe else "N/A"
|
exe = p_str(i("sys").executable) if track_exe else "N/A"
|
||||||
_id = hashlib.sha256((str(spec := [*pkgs]) + exe).encode()).hexdigest()
|
_id = i("hashlib").sha256((str(spec) + exe).encode()).hexdigest()
|
||||||
if (env := cache / (name if name else _id[:8])) not in cache.glob("*/") or F:
|
if (env := cache / (name if name else _id[:8])) not in cache.glob("*/") or F:
|
||||||
sys.stderr.write(f"generating new vivenv -> {env.name}\n")
|
i("sys").stderr.write(f"generating new vivenv -> {env.name}\n")
|
||||||
venv.create(env, prompt=f"viv-{name}", symlinks=True, clear=True)
|
i("venv").create(env, prompt=f"viv-{name}", symlinks=True, clear=True)
|
||||||
kw = dict(zip(("stdout", "stderr"), ((None,) * 2 if V else (-1, 2))))
|
run_kw = dict(zip(("stdout", "stderr"), ((None,) * 2 if V else (-1, 2))))
|
||||||
cmd = ["pip", "--python", str(env / "bin" / "python"), "install", *spec]
|
cmd = ["pip", "--python", str(env / "bin" / "python"), "install", *spec]
|
||||||
if (not no_st) and (not [x for x in spec if x.startswith("setuptools")]):
|
_if(ST, lambda: cmd.append("setuptools"))
|
||||||
cmd.append("setuptools")
|
if (rc := (p := i("subprocess").run(cmd, **run_kw)).returncode) != 0:
|
||||||
p = run(cmd, **kw)
|
_if(env.is_dir(), lambda: i("shutil").rmtree(env))
|
||||||
if (rc := p.returncode) != 0:
|
q(f"pip had non zero exit ({rc})\n{p.stdout.decode()}\n")
|
||||||
if env.is_dir():
|
add_meta(id=_id, spec=spec, exe=exe, name=name)
|
||||||
shutil.rmtree(env)
|
|
||||||
sys.stderr.write(f"pip had non zero exit ({rc})\n{p.stdout.decode()}\n")
|
|
||||||
sys.exit(rc)
|
|
||||||
meta.update(dict(id=_id, spec=spec, exe=exe, name=name, files=[runner]))
|
|
||||||
else:
|
else:
|
||||||
meta = json.loads((env / "vivmeta.json").read_text())
|
meta = i("json").loads((env / "vivmeta.json").read_text())
|
||||||
meta.update(dict(accessed=t, files=sorted({*meta["files"], runner})))
|
add_meta(accessed=now, files=sorted({*meta.get("files", []), p_str(__file__)}))
|
||||||
(env / "vivmeta.json").write_text(json.dumps(meta))
|
(env / "vivmeta.json").write_text(i("json").dumps(meta))
|
||||||
site.addsitedir(sitepkgs := str(*(env / "lib").glob("py*/si*")))
|
i("site").addsitedir(sitepkgs := str(*(env / "lib").glob("py*/si*")))
|
||||||
sys.path = [sitepkgs, *filter(nopkgs, sys.path)]
|
sys.path = [sitepkgs, *filter(nopkgs, sys.path)]
|
||||||
|
|
||||||
return env
|
return env
|
||||||
|
|
||||||
|
|
||||||
|
@ -50,11 +40,11 @@ import sys
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
vivenv = _viv_use(
|
vivenv = _viv_use(
|
||||||
"black==23.7.0",
|
"black==24.4.2",
|
||||||
"click==8.1.7",
|
"click==8.1.7",
|
||||||
"mypy-extensions==1.0.0",
|
"mypy-extensions==1.0.0",
|
||||||
"packaging==23.1", # noqa
|
"packaging==24.1", # noqa
|
||||||
"pathspec==0.11.2",
|
"pathspec==0.12.1",
|
||||||
"platformdirs==3.10.0",
|
"platformdirs==4.2.2",
|
||||||
) # noqa
|
) # noqa
|
||||||
sys.exit(subprocess.run([vivenv / "bin" / "black", *sys.argv[1:]]).returncode)
|
sys.exit(subprocess.run([vivenv / "bin" / "black", *sys.argv[1:]]).returncode)
|
||||||
|
|
|
@ -2008,44 +2008,34 @@ def err_quit(*msg: str, code: int = 1) -> NoReturn:
|
||||||
|
|
||||||
class Template:
|
class Template:
|
||||||
_standalone_func = r"""def _viv_use(*pkgs, track_exe=False, name=""):
|
_standalone_func = r"""def _viv_use(*pkgs, track_exe=False, name=""):
|
||||||
import hashlib, json, os, site, shutil, sys, venv # noqa
|
i, meta, add_meta = __import__, {}, lambda **kw: meta.update(**kw)
|
||||||
from pathlib import Path # noqa
|
P, ge, q, noop = i("pathlib").Path, i("os").getenv, i("sys").exit, lambda: None
|
||||||
from datetime import datetime # noqa
|
_if, p_str = lambda x, f: (noop, f)[x](), lambda x: f"{P(x).absolute().resolve()}"
|
||||||
from subprocess import run # noqa
|
|
||||||
|
|
||||||
if not {*map(type, pkgs)} == {str}:
|
|
||||||
raise ValueError(f"spec: {pkgs} is invalid")
|
|
||||||
|
|
||||||
meta = dict.fromkeys(("created", "accessed"), (t := str(datetime.today())))
|
|
||||||
runner = str(Path(__file__).absolute().resolve())
|
|
||||||
envvar = lambda x: os.getenv(f"VIV_{x}") # noqa
|
|
||||||
nopkgs = lambda p: not p.endswith(("dist-packages", "site-packages")) # noqa
|
nopkgs = lambda p: not p.endswith(("dist-packages", "site-packages")) # noqa
|
||||||
F, V, no_st = map(envvar, ("FORCE", "VERBOSE", "NO_SETUPTOOLS"))
|
_if({*map(type, spec := [*pkgs])} != {str}, lambda: q(f"spec: {pkgs} not string"))
|
||||||
base = Path(xdg) if (xdg := os.getenv("XDG_CACHE_HOME")) else Path.home() / ".cache"
|
F, V, NS = map(lambda x: ge(f"VIV_{x}"), ("FORCE", "VERBOSE", "NO_SETUPTOOLS"))
|
||||||
|
ST = not (NS and [x for x in spec if x.startswith("setuptools")])
|
||||||
|
add_meta(created=(now := str(i("datetime").datetime.today())))
|
||||||
|
base = P(xdg) if (xdg := ge("XDG_CACHE_HOME")) else P.home() / ".cache"
|
||||||
(cache := (base) / "viv/venvs").mkdir(parents=True, exist_ok=True)
|
(cache := (base) / "viv/venvs").mkdir(parents=True, exist_ok=True)
|
||||||
exe = str(Path(sys.executable).resolve()) if track_exe else "N/A"
|
exe = p_str(i("sys").executable) if track_exe else "N/A"
|
||||||
_id = hashlib.sha256((str(spec := [*pkgs]) + exe).encode()).hexdigest()
|
_id = i("hashlib").sha256((str(spec) + exe).encode()).hexdigest()
|
||||||
if (env := cache / (name if name else _id[:8])) not in cache.glob("*/") or F:
|
if (env := cache / (name if name else _id[:8])) not in cache.glob("*/") or F:
|
||||||
sys.stderr.write(f"generating new vivenv -> {env.name}\n")
|
i("sys").stderr.write(f"generating new vivenv -> {env.name}\n")
|
||||||
venv.create(env, prompt=f"viv-{name}", symlinks=True, clear=True)
|
i("venv").create(env, prompt=f"viv-{name}", symlinks=True, clear=True)
|
||||||
kw = dict(zip(("stdout", "stderr"), ((None,) * 2 if V else (-1, 2))))
|
run_kw = dict(zip(("stdout", "stderr"), ((None,) * 2 if V else (-1, 2))))
|
||||||
cmd = ["pip", "--python", str(env / "bin" / "python"), "install", *spec]
|
cmd = ["pip", "--python", str(env / "bin" / "python"), "install", *spec]
|
||||||
if (not no_st) and (not [x for x in spec if x.startswith("setuptools")]):
|
_if(ST, lambda: cmd.append("setuptools"))
|
||||||
cmd.append("setuptools")
|
if (rc := (p := i("subprocess").run(cmd, **run_kw)).returncode) != 0:
|
||||||
p = run(cmd, **kw)
|
_if(env.is_dir(), lambda: i("shutil").rmtree(env))
|
||||||
if (rc := p.returncode) != 0:
|
q(f"pip had non zero exit ({rc})\n{p.stdout.decode()}\n")
|
||||||
if env.is_dir():
|
add_meta(id=_id, spec=spec, exe=exe, name=name)
|
||||||
shutil.rmtree(env)
|
|
||||||
sys.stderr.write(f"pip had non zero exit ({rc})\n{p.stdout.decode()}\n")
|
|
||||||
sys.exit(rc)
|
|
||||||
meta.update(dict(id=_id, spec=spec, exe=exe, name=name, files=[runner]))
|
|
||||||
else:
|
else:
|
||||||
meta = json.loads((env / "vivmeta.json").read_text())
|
meta = i("json").loads((env / "vivmeta.json").read_text())
|
||||||
meta.update(dict(accessed=t, files=sorted({*meta["files"], runner})))
|
add_meta(accessed=now, files=sorted({*meta.get("files", []), p_str(__file__)}))
|
||||||
(env / "vivmeta.json").write_text(json.dumps(meta))
|
(env / "vivmeta.json").write_text(i("json").dumps(meta))
|
||||||
site.addsitedir(sitepkgs := str(*(env / "lib").glob("py*/si*")))
|
i("site").addsitedir(sitepkgs := str(*(env / "lib").glob("py*/si*")))
|
||||||
sys.path = [sitepkgs, *filter(nopkgs, sys.path)]
|
sys.path = [sitepkgs, *filter(nopkgs, sys.path)]
|
||||||
|
|
||||||
return env
|
return env
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
@ -3427,21 +3417,24 @@ class Viv:
|
||||||
cli: Path,
|
cli: Path,
|
||||||
yes: bool,
|
yes: bool,
|
||||||
) -> None:
|
) -> None:
|
||||||
to_remove = []
|
to_remove_dirs = []
|
||||||
|
to_remove_files = []
|
||||||
if Cfg().cache_base.is_dir():
|
if Cfg().cache_base.is_dir():
|
||||||
to_remove.append(Cfg().cache_base)
|
to_remove_dirs.append(Cfg().cache_base)
|
||||||
if src.is_file():
|
if src.is_file():
|
||||||
to_remove.append(src.parent if src == (Cfg().src) else src)
|
to_remove_files.append(src.parent if src == (Cfg().src) else src)
|
||||||
if self.local_source and self.local_source.is_file():
|
if self.local_source and self.local_source.is_file():
|
||||||
if self.local_source.parent.name == "viv":
|
if self.local_source.parent.name == "viv":
|
||||||
to_remove.append(self.local_source.parent)
|
to_remove_files.append(self.local_source.parent)
|
||||||
else:
|
else:
|
||||||
to_remove.append(self.local_source)
|
to_remove_files.append(self.local_source)
|
||||||
|
|
||||||
if cli.is_file():
|
if cli.is_file():
|
||||||
to_remove.append(cli)
|
to_remove_files.append(cli)
|
||||||
|
|
||||||
to_remove = list(set(to_remove))
|
to_remove = sorted(
|
||||||
|
set(to_remove_dirs + to_remove_files), key=lambda p: p.is_file()
|
||||||
|
)
|
||||||
if confirm(
|
if confirm(
|
||||||
"Remove the above files/directories?",
|
"Remove the above files/directories?",
|
||||||
"\n".join(f" - {a.red}{p}{a.end}" for p in to_remove) + "\n",
|
"\n".join(f" - {a.red}{p}{a.end}" for p in to_remove) + "\n",
|
||||||
|
@ -3449,6 +3442,7 @@ class Viv:
|
||||||
):
|
):
|
||||||
for p in to_remove:
|
for p in to_remove:
|
||||||
if p.is_dir():
|
if p.is_dir():
|
||||||
|
print(sorted(p.glob("**/*")))
|
||||||
shutil.rmtree(p)
|
shutil.rmtree(p)
|
||||||
else:
|
else:
|
||||||
p.unlink()
|
p.unlink()
|
||||||
|
|
|
@ -11,6 +11,7 @@ def test_use():
|
||||||
use("pyjokes")
|
use("pyjokes")
|
||||||
import pyjokes # noqa
|
import pyjokes # noqa
|
||||||
|
|
||||||
|
# sample is installed in the test venv which should be removed from sys.path
|
||||||
with pytest.raises(ImportError):
|
with pytest.raises(ImportError):
|
||||||
from sample.simple import add_one # noqa
|
from sample.simple import add_one # noqa
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue