-
Notifications
You must be signed in to change notification settings - Fork 7
/
get_venv.py
executable file
·92 lines (74 loc) · 2.82 KB
/
get_venv.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
#!/usr/bin/env python
"""Generate virtual environment activate shell command.
1. If venv/*/activate exists: print `source venv/*/activate`
2. elif it's a poetry project and not controlled by ssh: print `poetry shell`
3. elif .venv/*/activate exists: print `source .venv/*/activate`
4. else run `poetry env info --path` to get environment path then activated by `source ...`
"""
import functools
import os
import platform
def run_cmd(command):
# type: (str) -> str
with os.popen(command) as fp:
if not hasattr(fp, "_stream"): # For python2
return fp.read().strip()
bf = fp._stream.buffer.read().strip()
try:
return bf.decode()
except UnicodeDecodeError:
return bf.decode("gbk")
def read_content(filename):
# type: (str) -> bytes
if not os.path.exists(filename):
return b""
with open(filename, "rb") as f:
return f.read()
def is_controlled_by_ssh():
# type: () -> bool
return any(os.getenv(i) for i in "SSH_CLIENT SSH_TTY SSH_CONNECTION".split())
@functools.lru_cache
def is_poetry_project(filename):
# type: (str) -> bool
return b"[tool.poetry]" in read_content(filename)
def get_venv():
# type: () -> str
is_windows = platform.platform().lower().startswith("windows")
filename = "pyproject.toml"
common_venv_names = ["venv"]
if is_windows or not is_poetry_project(filename):
common_venv_names += [".venv"]
venv_dir = ""
for dirname in common_venv_names:
if os.path.exists(dirname):
venv_dir = dirname
break
else:
if (is_windows or is_controlled_by_ssh()) and is_poetry_project(filename):
# If use Git Bash at Windows, which does not show venv prefix after
# running `poetry shell`, should use `source ../activate` instead;
# When controlling by ssh in cloud server, `poetry shell` something
# cost 100% of CPU, sb got the similar issue in aws, and the `python-poetry`
# suggest to run `poetry run` or `source ../activate` to avoid it.
cache_dir = run_cmd("poetry env info --path")
if cache_dir:
try:
from pathlib import Path
except ImportError:
import sys
sys.exit(os.system("python3 " + sys.argv[0]))
venv_dir = Path(cache_dir.splitlines()[-1]).as_posix()
if venv_dir:
bin_dir = "*" if is_windows else "bin"
return "source {}/{}/activate".format(venv_dir, bin_dir)
elif is_poetry_project(filename):
return "poetry shell"
else:
# TODO: auto fix pdm
return "uv venv"
def main():
# type: () -> None
shell_command_to_activate_virtual_env = get_venv()
print(shell_command_to_activate_virtual_env)
if __name__ == "__main__":
main()