zrb 1.0.0a10__py3-none-any.whl → 1.0.0a12__py3-none-any.whl
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.
- zrb/builtin/__init__.py +15 -5
- zrb/builtin/setup/{dev → asdf}/asdf.py +6 -6
- zrb/builtin/setup/{system/latex → latex}/ubuntu.py +1 -1
- zrb/builtin/setup/{dev → tmux}/tmux.py +1 -1
- zrb/builtin/setup/tmux/tmux_config.sh +12 -0
- zrb/builtin/todo.py +84 -15
- zrb/config.py +1 -0
- zrb/input/base_input.py +1 -1
- zrb/input/bool_input.py +1 -1
- zrb/input/float_input.py +1 -1
- zrb/input/int_input.py +1 -1
- zrb/input/option_input.py +1 -1
- zrb/input/password_input.py +1 -1
- zrb/input/text_input.py +1 -1
- zrb/runner/web_app.py +27 -15
- zrb/util/load.py +4 -1
- zrb/util/todo.py +152 -34
- {zrb-1.0.0a10.dist-info → zrb-1.0.0a12.dist-info}/METADATA +1 -2
- {zrb-1.0.0a10.dist-info → zrb-1.0.0a12.dist-info}/RECORD +24 -24
- {zrb-1.0.0a10.dist-info → zrb-1.0.0a12.dist-info}/WHEEL +1 -1
- zrb/builtin/setup/dev/tmux_config.sh +0 -0
- /zrb/builtin/setup/{dev → asdf}/asdf_helper.py +0 -0
- /zrb/builtin/setup/{dev → tmux}/tmux_helper.py +0 -0
- /zrb/builtin/setup/{system/ubuntu.py → ubuntu.py} +0 -0
- {zrb-1.0.0a10.dist-info → zrb-1.0.0a12.dist-info}/entry_points.txt +0 -0
zrb/builtin/__init__.py
CHANGED
@@ -12,14 +12,22 @@ from zrb.builtin.md5 import hash_md5, sum_md5
|
|
12
12
|
from zrb.builtin.project.add.fastapp import add_fastapp_to_project
|
13
13
|
from zrb.builtin.project.create.create import create_project
|
14
14
|
from zrb.builtin.python import format_python_code
|
15
|
-
from zrb.builtin.setup.
|
16
|
-
from zrb.builtin.setup.
|
17
|
-
from zrb.builtin.setup.
|
18
|
-
from zrb.builtin.setup.
|
15
|
+
from zrb.builtin.setup.asdf.asdf import setup_asdf
|
16
|
+
from zrb.builtin.setup.latex.ubuntu import setup_latex_on_ubuntu
|
17
|
+
from zrb.builtin.setup.tmux.tmux import setup_tmux
|
18
|
+
from zrb.builtin.setup.ubuntu import setup_ubuntu
|
19
19
|
from zrb.builtin.shell.autocomplete.bash import make_bash_autocomplete
|
20
20
|
from zrb.builtin.shell.autocomplete.subcmd import get_shell_subcommands
|
21
21
|
from zrb.builtin.shell.autocomplete.zsh import make_zsh_autocomplete
|
22
|
-
from zrb.builtin.todo import
|
22
|
+
from zrb.builtin.todo import (
|
23
|
+
add_todo,
|
24
|
+
archive_todo,
|
25
|
+
complete_todo,
|
26
|
+
edit_todo,
|
27
|
+
list_todo,
|
28
|
+
log_todo,
|
29
|
+
show_todo,
|
30
|
+
)
|
23
31
|
|
24
32
|
assert create_project
|
25
33
|
assert add_fastapp_to_project
|
@@ -42,9 +50,11 @@ assert git_pull_subtree
|
|
42
50
|
assert git_push_subtree
|
43
51
|
assert list_todo
|
44
52
|
assert add_todo
|
53
|
+
assert archive_todo
|
45
54
|
assert edit_todo
|
46
55
|
assert complete_todo
|
47
56
|
assert log_todo
|
57
|
+
assert show_todo
|
48
58
|
assert setup_ubuntu
|
49
59
|
assert setup_latex_on_ubuntu
|
50
60
|
assert setup_asdf
|
@@ -1,6 +1,12 @@
|
|
1
1
|
import os
|
2
2
|
|
3
3
|
from zrb.builtin.group import setup_group
|
4
|
+
from zrb.builtin.setup.asdf.asdf_helper import (
|
5
|
+
check_inexist_asdf_dir,
|
6
|
+
get_install_prerequisites_cmd,
|
7
|
+
setup_asdf_ps_config,
|
8
|
+
setup_asdf_sh_config,
|
9
|
+
)
|
4
10
|
from zrb.builtin.setup.common_input import (
|
5
11
|
package_manager_input,
|
6
12
|
setup_bash_input,
|
@@ -8,12 +14,6 @@ from zrb.builtin.setup.common_input import (
|
|
8
14
|
setup_zsh_input,
|
9
15
|
use_sudo_input,
|
10
16
|
)
|
11
|
-
from zrb.builtin.setup.dev.asdf_helper import (
|
12
|
-
check_inexist_asdf_dir,
|
13
|
-
get_install_prerequisites_cmd,
|
14
|
-
setup_asdf_ps_config,
|
15
|
-
setup_asdf_sh_config,
|
16
|
-
)
|
17
17
|
from zrb.context.any_context import AnyContext
|
18
18
|
from zrb.task.cmd_task import CmdTask
|
19
19
|
from zrb.task.make_task import make_task
|
@@ -2,7 +2,7 @@ import os
|
|
2
2
|
|
3
3
|
from zrb.builtin.group import setup_group
|
4
4
|
from zrb.builtin.setup.common_input import package_manager_input, use_sudo_input
|
5
|
-
from zrb.builtin.setup.
|
5
|
+
from zrb.builtin.setup.tmux.tmux_helper import get_install_tmux_cmd
|
6
6
|
from zrb.context.any_context import AnyContext
|
7
7
|
from zrb.input.str_input import StrInput
|
8
8
|
from zrb.task.cmd_task import CmdTask
|
@@ -0,0 +1,12 @@
|
|
1
|
+
set -g @plugin 'tmux-plugins/tpm'
|
2
|
+
set -g @plugin 'tmux-plugins/tmux-sensible'
|
3
|
+
run '~/.tmux/plugins/tpm/tpm'
|
4
|
+
|
5
|
+
set-option -sg escape-time 10
|
6
|
+
set-option -g focus-events On
|
7
|
+
set-option -g default-terminal "screen-256color"
|
8
|
+
set-option -sa terminal-overrides ',xterm-256color:RGB'
|
9
|
+
|
10
|
+
bind c new-window -c "#{pane_current_path}"
|
11
|
+
bind '"' split-window -c "#{pane_current_path}"
|
12
|
+
bind % split-window -h -c "#{pane_current_path}"
|
zrb/builtin/todo.py
CHANGED
@@ -4,7 +4,7 @@ import os
|
|
4
4
|
from typing import Any
|
5
5
|
|
6
6
|
from zrb.builtin.group import todo_group
|
7
|
-
from zrb.config import TODO_DIR
|
7
|
+
from zrb.config import TODO_DIR, TODO_VISUAL_FILTER
|
8
8
|
from zrb.context.any_context import AnyContext
|
9
9
|
from zrb.input.str_input import StrInput
|
10
10
|
from zrb.input.text_input import TextInput
|
@@ -13,6 +13,7 @@ from zrb.util.todo import (
|
|
13
13
|
TodoTaskModel,
|
14
14
|
add_durations,
|
15
15
|
cascade_todo_task,
|
16
|
+
get_visual_todo_card,
|
16
17
|
get_visual_todo_list,
|
17
18
|
line_to_todo_task,
|
18
19
|
load_todo_list,
|
@@ -23,7 +24,7 @@ from zrb.util.todo import (
|
|
23
24
|
|
24
25
|
|
25
26
|
@make_task(
|
26
|
-
name="todo
|
27
|
+
name="add-todo",
|
27
28
|
input=[
|
28
29
|
StrInput(
|
29
30
|
name="description",
|
@@ -77,20 +78,52 @@ def add_todo(ctx: AnyContext):
|
|
77
78
|
)
|
78
79
|
)
|
79
80
|
save_todo_list(todo_file_path, todo_list)
|
80
|
-
return get_visual_todo_list(todo_list)
|
81
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
81
82
|
|
82
83
|
|
83
|
-
@make_task(name="todo
|
84
|
+
@make_task(name="list-todo", description="📋 List todo", group=todo_group, alias="list")
|
84
85
|
def list_todo(ctx: AnyContext):
|
85
86
|
todo_file_path = os.path.join(TODO_DIR, "todo.txt")
|
86
|
-
|
87
|
+
todo_list: list[TodoTaskModel] = []
|
87
88
|
if os.path.isfile(todo_file_path):
|
88
|
-
|
89
|
-
return get_visual_todo_list(
|
89
|
+
todo_list = load_todo_list(todo_file_path)
|
90
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
90
91
|
|
91
92
|
|
92
93
|
@make_task(
|
93
|
-
name="todo
|
94
|
+
name="show-todo",
|
95
|
+
input=StrInput(name="keyword", prompt="Task keyword", description="Task Keyword"),
|
96
|
+
description="🔍 Show todo",
|
97
|
+
group=todo_group,
|
98
|
+
alias="show",
|
99
|
+
)
|
100
|
+
def show_todo(ctx: AnyContext):
|
101
|
+
todo_file_path = os.path.join(TODO_DIR, "todo.txt")
|
102
|
+
todo_list: list[TodoTaskModel] = []
|
103
|
+
todo_list: list[TodoTaskModel] = []
|
104
|
+
if os.path.isfile(todo_file_path):
|
105
|
+
todo_list = load_todo_list(todo_file_path)
|
106
|
+
# Get todo task
|
107
|
+
todo_task = select_todo_task(todo_list, ctx.input.keyword)
|
108
|
+
if todo_task is None:
|
109
|
+
ctx.log_error("Task not found")
|
110
|
+
return
|
111
|
+
if todo_task.completed:
|
112
|
+
ctx.log_error("Task already completed")
|
113
|
+
return
|
114
|
+
# Update todo task
|
115
|
+
todo_task = cascade_todo_task(todo_task)
|
116
|
+
task_id = todo_task.keyval.get("id", "")
|
117
|
+
log_work_path = os.path.join(TODO_DIR, "log-work", f"{task_id}.json")
|
118
|
+
log_work_list = []
|
119
|
+
if os.path.isfile(log_work_path):
|
120
|
+
with open(log_work_path, "r") as f:
|
121
|
+
log_work_list = json.loads(f.read())
|
122
|
+
return get_visual_todo_card(todo_task, log_work_list)
|
123
|
+
|
124
|
+
|
125
|
+
@make_task(
|
126
|
+
name="complete-todo",
|
94
127
|
input=StrInput(name="keyword", prompt="Task keyword", description="Task Keyword"),
|
95
128
|
description="✅ Complete todo",
|
96
129
|
group=todo_group,
|
@@ -105,7 +138,10 @@ def complete_todo(ctx: AnyContext):
|
|
105
138
|
todo_task = select_todo_task(todo_list, ctx.input.keyword)
|
106
139
|
if todo_task is None:
|
107
140
|
ctx.log_error("Task not found")
|
108
|
-
return get_visual_todo_list(todo_list)
|
141
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
142
|
+
if todo_task.completed:
|
143
|
+
ctx.log_error("Task already completed")
|
144
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
109
145
|
# Update todo task
|
110
146
|
todo_task = cascade_todo_task(todo_task)
|
111
147
|
if todo_task.creation_date is not None:
|
@@ -113,11 +149,43 @@ def complete_todo(ctx: AnyContext):
|
|
113
149
|
todo_task.completed = True
|
114
150
|
# Save todo list
|
115
151
|
save_todo_list(todo_file_path, todo_list)
|
116
|
-
return get_visual_todo_list(todo_list)
|
152
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
117
153
|
|
118
154
|
|
119
155
|
@make_task(
|
120
|
-
name="todo
|
156
|
+
name="archive-todo",
|
157
|
+
description="📚 Archive todo",
|
158
|
+
group=todo_group,
|
159
|
+
alias="archive",
|
160
|
+
)
|
161
|
+
def archive_todo(ctx: AnyContext):
|
162
|
+
todo_file_path = os.path.join(TODO_DIR, "todo.txt")
|
163
|
+
todo_list: list[TodoTaskModel] = []
|
164
|
+
if os.path.isfile(todo_file_path):
|
165
|
+
todo_list = load_todo_list(todo_file_path)
|
166
|
+
working_todo_list = [
|
167
|
+
todo_task for todo_task in todo_list if not todo_task.completed
|
168
|
+
]
|
169
|
+
new_archived_todo_list = [
|
170
|
+
todo_task for todo_task in todo_list if todo_task.completed
|
171
|
+
]
|
172
|
+
if len(new_archived_todo_list) == 0:
|
173
|
+
ctx.print("No completed task to archive")
|
174
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
175
|
+
archive_file_path = os.path.join(TODO_DIR, "archive.txt")
|
176
|
+
if not os.path.isdir(TODO_DIR):
|
177
|
+
os.make_dirs(TODO_DIR, exist_ok=True)
|
178
|
+
archived_todo_list = []
|
179
|
+
if os.path.isfile(archive_file_path):
|
180
|
+
archived_todo_list = load_todo_list(archive_file_path)
|
181
|
+
archived_todo_list += new_archived_todo_list
|
182
|
+
save_todo_list(archive_file_path, archived_todo_list)
|
183
|
+
save_todo_list(todo_file_path, working_todo_list)
|
184
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
185
|
+
|
186
|
+
|
187
|
+
@make_task(
|
188
|
+
name="log-todo",
|
121
189
|
input=[
|
122
190
|
StrInput(name="keyword", prompt="Task keyword", description="Task Keyword"),
|
123
191
|
StrInput(
|
@@ -151,7 +219,7 @@ def log_todo(ctx: AnyContext):
|
|
151
219
|
todo_task = select_todo_task(todo_list, ctx.input.keyword)
|
152
220
|
if todo_task is None:
|
153
221
|
ctx.log_error("Task not found")
|
154
|
-
return get_visual_todo_list(todo_list)
|
222
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
155
223
|
# Update todo task
|
156
224
|
todo_task = cascade_todo_task(todo_task)
|
157
225
|
current_duration = todo_task.keyval.get("duration", "0")
|
@@ -175,7 +243,7 @@ def log_todo(ctx: AnyContext):
|
|
175
243
|
)
|
176
244
|
with open(log_work_file_path, "w") as f:
|
177
245
|
f.write(json.dumps(log_work, indent=2))
|
178
|
-
return get_visual_todo_list(todo_list)
|
246
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
179
247
|
|
180
248
|
|
181
249
|
def _get_default_start() -> str:
|
@@ -183,7 +251,7 @@ def _get_default_start() -> str:
|
|
183
251
|
|
184
252
|
|
185
253
|
@make_task(
|
186
|
-
name="todo
|
254
|
+
name="edit-todo",
|
187
255
|
input=[
|
188
256
|
TextInput(
|
189
257
|
name="text",
|
@@ -207,7 +275,7 @@ def edit_todo(ctx: AnyContext):
|
|
207
275
|
with open(todo_file_path, "w") as f:
|
208
276
|
f.write(new_content)
|
209
277
|
todo_list = load_todo_list(todo_file_path)
|
210
|
-
return get_visual_todo_list(todo_list)
|
278
|
+
return get_visual_todo_list(todo_list, TODO_VISUAL_FILTER)
|
211
279
|
|
212
280
|
|
213
281
|
def _get_todo_txt_content() -> str:
|
@@ -216,3 +284,4 @@ def _get_todo_txt_content() -> str:
|
|
216
284
|
return ""
|
217
285
|
with open(todo_file_path, "r") as f:
|
218
286
|
return f.read()
|
287
|
+
return f.read()
|
zrb/config.py
CHANGED
@@ -56,6 +56,7 @@ SESSION_LOG_DIR = os.getenv(
|
|
56
56
|
"ZRB_SESSION_LOG_DIR", os.path.expanduser(os.path.join("~", ".zrb-session"))
|
57
57
|
)
|
58
58
|
TODO_DIR = os.getenv("ZRB_TODO_DIR", os.path.expanduser(os.path.join("~", "todo")))
|
59
|
+
TODO_VISUAL_FILTER = os.getenv("ZRB_TODO_FILTER", "")
|
59
60
|
VERSION = metadata.version("zrb")
|
60
61
|
WEB_HTTP_PORT = int(os.getenv("ZRB_WEB_HTTP_PORT", "21213"))
|
61
62
|
LLM_MODEL = os.getenv("ZRB_LLM_MODEL", "ollama_chat/llama3.1")
|
zrb/input/base_input.py
CHANGED
zrb/input/bool_input.py
CHANGED
zrb/input/float_input.py
CHANGED
zrb/input/int_input.py
CHANGED
zrb/input/option_input.py
CHANGED
zrb/input/password_input.py
CHANGED
zrb/input/text_input.py
CHANGED
@@ -16,7 +16,7 @@ class TextInput(BaseInput):
|
|
16
16
|
prompt: str | None = None,
|
17
17
|
default_str: str | Callable[[AnySharedContext], str] = "",
|
18
18
|
auto_render: bool = True,
|
19
|
-
allow_empty: bool =
|
19
|
+
allow_empty: bool = False,
|
20
20
|
editor: str = DEFAULT_EDITOR,
|
21
21
|
extension: str = ".txt",
|
22
22
|
comment_start: str | None = None,
|
zrb/runner/web_app.py
CHANGED
@@ -2,7 +2,7 @@ import asyncio
|
|
2
2
|
import os
|
3
3
|
import sys
|
4
4
|
from datetime import datetime, timedelta
|
5
|
-
from typing import Any
|
5
|
+
from typing import Any
|
6
6
|
|
7
7
|
from zrb.config import BANNER, WEB_HTTP_PORT
|
8
8
|
from zrb.context.shared_context import SharedContext
|
@@ -23,7 +23,7 @@ from zrb.util.group import extract_node_from_args, get_node_path
|
|
23
23
|
def create_app(root_group: AnyGroup, port: int = WEB_HTTP_PORT):
|
24
24
|
from contextlib import asynccontextmanager
|
25
25
|
|
26
|
-
from fastapi import FastAPI, HTTPException, Request
|
26
|
+
from fastapi import FastAPI, HTTPException, Query, Request
|
27
27
|
from fastapi.responses import FileResponse, HTMLResponse
|
28
28
|
from fastapi.staticfiles import StaticFiles
|
29
29
|
|
@@ -97,7 +97,13 @@ def create_app(root_group: AnyGroup, port: int = WEB_HTTP_PORT):
|
|
97
97
|
raise HTTPException(status_code=404, detail="Not Found")
|
98
98
|
|
99
99
|
@app.get("/api/{path:path}", response_model=SessionStateLog | SessionStateLogList)
|
100
|
-
async def get_session(
|
100
|
+
async def get_session(
|
101
|
+
path: str,
|
102
|
+
min_start_query: str = Query(default=None, alias="from"),
|
103
|
+
max_start_query: str = Query(default=None, alias="to"),
|
104
|
+
page: int = Query(default=0, alias="page"),
|
105
|
+
limit: int = Query(default=10, alias="limit"),
|
106
|
+
):
|
101
107
|
"""
|
102
108
|
Getting existing session or sessions
|
103
109
|
"""
|
@@ -106,24 +112,30 @@ def create_app(root_group: AnyGroup, port: int = WEB_HTTP_PORT):
|
|
106
112
|
if isinstance(node, AnyTask) and residual_args:
|
107
113
|
if residual_args[0] == "list":
|
108
114
|
task_path = get_node_path(root_group, node)
|
109
|
-
|
115
|
+
max_start_time = (
|
116
|
+
datetime.now()
|
117
|
+
if max_start_query is None
|
118
|
+
else datetime.strptime(max_start_query, "%Y-%m-%d %H:%M:%S")
|
119
|
+
)
|
120
|
+
min_start_time = (
|
121
|
+
max_start_time - timedelta(hours=1)
|
122
|
+
if min_start_query is None
|
123
|
+
else datetime.strptime(min_start_query, "%Y-%m-%d %H:%M:%S")
|
124
|
+
)
|
125
|
+
return list_sessions(
|
126
|
+
task_path, min_start_time, max_start_time, page, limit
|
127
|
+
)
|
110
128
|
else:
|
111
129
|
return read_session(residual_args[0])
|
112
130
|
raise HTTPException(status_code=404, detail="Not Found")
|
113
131
|
|
114
132
|
def list_sessions(
|
115
|
-
task_path:
|
133
|
+
task_path: list[str],
|
134
|
+
min_start_time: datetime,
|
135
|
+
max_start_time: datetime,
|
136
|
+
page: int,
|
137
|
+
limit: int,
|
116
138
|
) -> SessionStateLogList:
|
117
|
-
max_start_time = datetime.now()
|
118
|
-
if "to" in query_params:
|
119
|
-
max_start_time = datetime.strptime(query_params["to"], "%Y-%m-%d %H:%M:%S")
|
120
|
-
min_start_time = max_start_time - timedelta(hours=1)
|
121
|
-
if "from" in query_params:
|
122
|
-
min_start_time = datetime.strptime(
|
123
|
-
query_params["from"], "%Y-%m-%d %H:%M:%S"
|
124
|
-
)
|
125
|
-
page = int(query_params.get("page", 0))
|
126
|
-
limit = int(query_params.get("limit", 10))
|
127
139
|
try:
|
128
140
|
return default_session_state_logger.list(
|
129
141
|
task_path,
|
zrb/util/load.py
CHANGED
@@ -3,6 +3,7 @@ import importlib.util
|
|
3
3
|
import os
|
4
4
|
import re
|
5
5
|
import sys
|
6
|
+
from functools import lru_cache
|
6
7
|
from typing import Any
|
7
8
|
|
8
9
|
pattern = re.compile("[^a-zA-Z0-9]")
|
@@ -13,7 +14,7 @@ def load_zrb_init(dir_path: str | None = None):
|
|
13
14
|
dir_path = os.getcwd()
|
14
15
|
script_path = os.path.join(dir_path, "zrb_init.py")
|
15
16
|
if os.path.isfile(script_path):
|
16
|
-
load_file(script_path)
|
17
|
+
load_file(script_path, -1)
|
17
18
|
return
|
18
19
|
new_dir_path = os.path.dirname(dir_path)
|
19
20
|
if new_dir_path == dir_path:
|
@@ -21,6 +22,7 @@ def load_zrb_init(dir_path: str | None = None):
|
|
21
22
|
load_zrb_init(new_dir_path)
|
22
23
|
|
23
24
|
|
25
|
+
@lru_cache()
|
24
26
|
def load_file(script_path: str, sys_path_index: int = 0):
|
25
27
|
if not os.path.isfile(script_path):
|
26
28
|
return
|
@@ -45,6 +47,7 @@ def _get_new_python_path(dir_path: str) -> str:
|
|
45
47
|
return ":".join([current_python_path, dir_path])
|
46
48
|
|
47
49
|
|
50
|
+
@lru_cache()
|
48
51
|
def load_module(module_name: str) -> Any:
|
49
52
|
module = importlib.import_module(module_name)
|
50
53
|
return module
|
zrb/util/todo.py
CHANGED
@@ -1,11 +1,15 @@
|
|
1
1
|
import datetime
|
2
2
|
import re
|
3
|
+
import shutil
|
4
|
+
from typing import Any
|
3
5
|
|
4
6
|
from pydantic import BaseModel, Field, model_validator
|
5
7
|
|
6
8
|
from zrb.util.cli.style import (
|
7
9
|
stylize_bold_green,
|
10
|
+
stylize_bold_yellow,
|
8
11
|
stylize_cyan,
|
12
|
+
stylize_faint,
|
9
13
|
stylize_magenta,
|
10
14
|
stylize_yellow,
|
11
15
|
)
|
@@ -32,6 +36,16 @@ class TodoTaskModel(BaseModel):
|
|
32
36
|
)
|
33
37
|
return values
|
34
38
|
|
39
|
+
def get_additional_info_length(self):
|
40
|
+
results = []
|
41
|
+
for project in self.projects:
|
42
|
+
results.append(f"@{project}")
|
43
|
+
for context in self.contexts:
|
44
|
+
results.append(f"+{context}")
|
45
|
+
for key, val in self.keyval.items():
|
46
|
+
results.append(f"{key}:{val}")
|
47
|
+
return len(", ".join(results))
|
48
|
+
|
35
49
|
|
36
50
|
TODO_TXT_PATTERN = re.compile(
|
37
51
|
r"^(?P<status>x)?\s*" # Optional completion mark ('x')
|
@@ -80,7 +94,8 @@ def load_todo_list(todo_file_path: str) -> list[TodoTaskModel]:
|
|
80
94
|
todo_line = todo_line.strip()
|
81
95
|
if todo_line == "":
|
82
96
|
continue
|
83
|
-
|
97
|
+
todo_task = line_to_todo_task(todo_line)
|
98
|
+
todo_list.append(todo_task)
|
84
99
|
todo_list.sort(
|
85
100
|
key=lambda task: (
|
86
101
|
task.completed,
|
@@ -173,51 +188,154 @@ def todo_task_to_line(task: TodoTaskModel) -> str:
|
|
173
188
|
return " ".join(parts)
|
174
189
|
|
175
190
|
|
176
|
-
def get_visual_todo_list(todo_list: list[TodoTaskModel]) -> str:
|
177
|
-
|
191
|
+
def get_visual_todo_list(todo_list: list[TodoTaskModel], filter: str) -> str:
|
192
|
+
todo_filter = line_to_todo_task(filter)
|
193
|
+
filtered_todo_list = []
|
194
|
+
for todo_task in todo_list:
|
195
|
+
filter_description = todo_filter.description.lower().strip()
|
196
|
+
if (
|
197
|
+
filter_description != ""
|
198
|
+
and filter_description not in todo_task.description.lower()
|
199
|
+
):
|
200
|
+
continue
|
201
|
+
if not all(context in todo_task.contexts for context in todo_filter.contexts):
|
202
|
+
continue
|
203
|
+
if not all(project in todo_task.projects for project in todo_filter.projects):
|
204
|
+
continue
|
205
|
+
if not all(
|
206
|
+
key in todo_task.keyval and todo_task.keyval[key] == val
|
207
|
+
for key, val in todo_filter.keyval.items()
|
208
|
+
):
|
209
|
+
continue
|
210
|
+
filtered_todo_list.append(todo_task)
|
211
|
+
if len(filtered_todo_list) == 0:
|
178
212
|
return "\n".join(["", " Empty todo list... 🌵🦖", ""])
|
179
|
-
|
180
|
-
|
181
|
-
|
213
|
+
max_desc_length = max(
|
214
|
+
len(todo_task.description) for todo_task in filtered_todo_list
|
215
|
+
)
|
216
|
+
if max_desc_length < len("DESCRIPTION"):
|
217
|
+
max_desc_length = len("DESCRIPTION")
|
218
|
+
if max_desc_length > 70:
|
219
|
+
max_desc_length = 70
|
220
|
+
max_additional_info_length = max(
|
221
|
+
todo_task.get_additional_info_length() for todo_task in filtered_todo_list
|
222
|
+
)
|
223
|
+
if max_additional_info_length < len("PROJECT/CONTEXT/OTHERS"):
|
224
|
+
max_additional_info_length = len("PROJECT/CONTEXT/OTHERS")
|
225
|
+
terminal_width, _ = shutil.get_terminal_size()
|
182
226
|
# Headers
|
183
227
|
results = [
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
"".ljust(3), # priority
|
188
|
-
"".ljust(3), # completed
|
189
|
-
"COMPLETED AT".rjust(14), # completed date
|
190
|
-
"CREATED AT".rjust(14), # completed date
|
191
|
-
"DESCRIPTION".ljust(max_desc_name_length),
|
192
|
-
"PROJECT/CONTEXT/OTHERS",
|
193
|
-
]
|
228
|
+
stylize_faint(
|
229
|
+
get_visual_todo_header(
|
230
|
+
terminal_width, max_desc_length, max_additional_info_length
|
194
231
|
)
|
195
232
|
)
|
196
233
|
]
|
197
|
-
for todo_task in
|
198
|
-
completed = "[x]" if todo_task.completed else "[ ]"
|
199
|
-
priority = " " if todo_task.priority is None else f"({todo_task.priority})"
|
200
|
-
completion_date = stylize_yellow(_date_to_str(todo_task.completion_date))
|
201
|
-
creation_date = stylize_cyan(_date_to_str(todo_task.creation_date))
|
202
|
-
description = todo_task.description.ljust(max_desc_name_length)
|
203
|
-
additions = ", ".join(
|
204
|
-
[stylize_yellow(f"+{project}") for project in todo_task.projects]
|
205
|
-
+ [stylize_cyan(f"@{context}") for context in todo_task.contexts]
|
206
|
-
+ [stylize_magenta(f"{key}:{val}") for key, val in todo_task.keyval.items()]
|
207
|
-
)
|
234
|
+
for todo_task in filtered_todo_list:
|
208
235
|
results.append(
|
236
|
+
get_visual_todo_line(
|
237
|
+
terminal_width, max_desc_length, max_additional_info_length, todo_task
|
238
|
+
)
|
239
|
+
)
|
240
|
+
return "\n".join(results)
|
241
|
+
|
242
|
+
|
243
|
+
def get_visual_todo_header(
|
244
|
+
terminal_width: int, max_desc_length: int, max_additional_info_length: int
|
245
|
+
) -> str:
|
246
|
+
priority = "".ljust(3)
|
247
|
+
completed = "".ljust(3)
|
248
|
+
completed_at = "COMPLETED AT".rjust(14)
|
249
|
+
created_at = "CREATED_AT".ljust(14)
|
250
|
+
description = "DESCRIPTION".ljust(min(max_desc_length, 70))
|
251
|
+
additional_info = "PROJECT/CONTEXT/OTHERS"
|
252
|
+
if terminal_width <= 14 + max_desc_length + max_additional_info_length:
|
253
|
+
return " ".join([priority, completed, description])
|
254
|
+
if terminal_width <= 36 + max_desc_length + max_additional_info_length:
|
255
|
+
return " ".join([priority, completed, description, additional_info])
|
256
|
+
return " ".join(
|
257
|
+
[priority, completed, completed_at, created_at, description, additional_info]
|
258
|
+
)
|
259
|
+
|
260
|
+
|
261
|
+
def get_visual_todo_line(
|
262
|
+
terminal_width: int,
|
263
|
+
max_desc_length: int,
|
264
|
+
max_additional_info_length: int,
|
265
|
+
todo_task: TodoTaskModel,
|
266
|
+
) -> str:
|
267
|
+
completed = "[x]" if todo_task.completed else "[ ]"
|
268
|
+
priority = " " if todo_task.priority is None else f"({todo_task.priority})"
|
269
|
+
completed_at = stylize_yellow(_date_to_str(todo_task.completion_date))
|
270
|
+
created_at = stylize_cyan(_date_to_str(todo_task.creation_date))
|
271
|
+
description = todo_task.description
|
272
|
+
if len(description) > max_desc_length:
|
273
|
+
description = description[: max_desc_length - 4] + " ..."
|
274
|
+
description = description.ljust(max_desc_length)
|
275
|
+
description = description[:max_desc_length]
|
276
|
+
if todo_task.completed:
|
277
|
+
description = stylize_faint(description)
|
278
|
+
elif "duration" in todo_task.keyval:
|
279
|
+
description = stylize_bold_yellow(description)
|
280
|
+
additional_info = ", ".join(
|
281
|
+
[stylize_yellow(f"+{project}") for project in todo_task.projects]
|
282
|
+
+ [stylize_cyan(f"@{context}") for context in todo_task.contexts]
|
283
|
+
+ [stylize_magenta(f"{key}:{val}") for key, val in todo_task.keyval.items()]
|
284
|
+
)
|
285
|
+
if terminal_width <= 14 + max_desc_length + max_additional_info_length:
|
286
|
+
return " ".join([priority, completed, description])
|
287
|
+
if terminal_width <= 36 + max_desc_length + max_additional_info_length:
|
288
|
+
return " ".join([priority, completed, description, additional_info])
|
289
|
+
return " ".join(
|
290
|
+
[priority, completed, completed_at, created_at, description, additional_info]
|
291
|
+
)
|
292
|
+
|
293
|
+
|
294
|
+
def get_visual_todo_card(
|
295
|
+
todo_task: TodoTaskModel, log_work_list: list[dict[str, str]]
|
296
|
+
) -> str:
|
297
|
+
description = todo_task.description
|
298
|
+
status = "TODO"
|
299
|
+
if todo_task.completed:
|
300
|
+
status = "DONE"
|
301
|
+
elif "duration" in todo_task.keyval:
|
302
|
+
status = "DOING"
|
303
|
+
priority = todo_task.priority
|
304
|
+
completed_at = (
|
305
|
+
_date_to_str(todo_task.completion_date)
|
306
|
+
if todo_task.completion_date is not None
|
307
|
+
else ""
|
308
|
+
)
|
309
|
+
created_at = (
|
310
|
+
_date_to_str(todo_task.creation_date)
|
311
|
+
if todo_task.creation_date is not None
|
312
|
+
else ""
|
313
|
+
)
|
314
|
+
log_work_str = "\n".join(
|
315
|
+
[
|
209
316
|
" ".join(
|
210
317
|
[
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
-
creation_date,
|
215
|
-
description,
|
216
|
-
additions,
|
318
|
+
log_work.get("duration", "").strip().rjust(12),
|
319
|
+
log_work.get("start", "").strip().rjust(20),
|
320
|
+
log_work.get("log", "").strip(),
|
217
321
|
]
|
218
322
|
)
|
323
|
+
for log_work in log_work_list
|
324
|
+
]
|
325
|
+
)
|
326
|
+
detail = [
|
327
|
+
f"{'📄 Description'.ljust(16)}: {description}",
|
328
|
+
f"{'🎯 Priority'.ljust(16)}: {priority}",
|
329
|
+
f"{'📊 Status'.ljust(16)}: {status}",
|
330
|
+
f"{'📅 Created at'.ljust(16)}: {created_at}",
|
331
|
+
f"{'✅ Completed at'.ljust(16)}: {completed_at}",
|
332
|
+
]
|
333
|
+
if log_work_str != "":
|
334
|
+
detail.append(
|
335
|
+
stylize_faint(" ".join(["Time Spent".rjust(12), "Start".rjust(20), "Log"]))
|
219
336
|
)
|
220
|
-
|
337
|
+
detail.append(log_work_str)
|
338
|
+
return "\n".join(detail)
|
221
339
|
|
222
340
|
|
223
341
|
def _date_to_str(date: datetime.date | None) -> str:
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.1
|
2
2
|
Name: zrb
|
3
|
-
Version: 1.0.
|
3
|
+
Version: 1.0.0a12
|
4
4
|
Summary: Your Automation Powerhouse
|
5
5
|
Home-page: https://github.com/state-alchemists/zrb
|
6
6
|
License: AGPL-3.0-or-later
|
@@ -13,7 +13,6 @@ Classifier: Programming Language :: Python :: 3
|
|
13
13
|
Classifier: Programming Language :: Python :: 3.10
|
14
14
|
Classifier: Programming Language :: Python :: 3.11
|
15
15
|
Classifier: Programming Language :: Python :: 3.12
|
16
|
-
Classifier: Programming Language :: Python :: 3.13
|
17
16
|
Provides-Extra: rag
|
18
17
|
Requires-Dist: autopep8 (>=2.0.4,<3.0.0)
|
19
18
|
Requires-Dist: beautifulsoup4 (>=4.12.3,<5.0.0)
|
@@ -2,7 +2,7 @@ zrb/__init__.py,sha256=ESskletjBpLO78a6wLigi9wnEGtmhfc3i_rbyMvrSyU,2767
|
|
2
2
|
zrb/__main__.py,sha256=aiIGigf24Yh4B8KVvJxxAcjvb_CSc13yfpeB6EkRHhE,645
|
3
3
|
zrb/attr/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
4
4
|
zrb/attr/type.py,sha256=4TV5gPYMMrKh5V-yB6iRYKCbsXAH_AvGXMsjxKLHcUs,568
|
5
|
-
zrb/builtin/__init__.py,sha256=
|
5
|
+
zrb/builtin/__init__.py,sha256=seqFeTQk4k9I31oTdVrCDoud4_AzCrdVEZx1VxVUtZk,1744
|
6
6
|
zrb/builtin/base64.py,sha256=1YnSwASp7OEAvQcsnHZGpJEvYoI1Z2zTIJ1bCDHfcPQ,921
|
7
7
|
zrb/builtin/git.py,sha256=ypxGT4JyokYbXMvxdWFG684sEa6x7wht1b4PxpddTMA,4544
|
8
8
|
zrb/builtin/git_subtree.py,sha256=M5Fr8NB4LaC3ra997tzZMLeD4H5o84MvvwefrLcWEIk,3016
|
@@ -69,27 +69,27 @@ zrb/builtin/project/create/create.py,sha256=IUU-4oUlESiw-ZHtnkbhnEA62_thKMDwGdAq
|
|
69
69
|
zrb/builtin/project/create/project-template/README.md,sha256=BHeWF_txdTDzKewWdWfGhO3YOTiW8E3YwTMnd0T1LVA,39
|
70
70
|
zrb/builtin/project/create/project-template/zrb_init.py,sha256=cbPl2xZbr1LXBGd296yCvfullzmryG5nTZ8mcBmu_vU,89
|
71
71
|
zrb/builtin/python.py,sha256=herHFZl5wHsMFcqHhpisg9Ql3pQXqoCuTTwzNUTXzkw,281
|
72
|
+
zrb/builtin/setup/asdf/asdf.py,sha256=n_dWoyYa7vZ9tBCy8hgxmSWZ-XHltC1R70mIVZQhnXY,2419
|
73
|
+
zrb/builtin/setup/asdf/asdf_helper.py,sha256=804vfpsDPH3bpOmpAZyGxJdtwcmxNvuj9i0vaIW2qNI,1318
|
72
74
|
zrb/builtin/setup/common_input.py,sha256=M8q9Unt7dCfBHTHqV3__Mxk9YC2Pl3DDtxat2BanZCQ,870
|
73
|
-
zrb/builtin/setup/
|
74
|
-
zrb/builtin/setup/
|
75
|
-
zrb/builtin/setup/
|
76
|
-
zrb/builtin/setup/
|
77
|
-
zrb/builtin/setup/
|
78
|
-
zrb/builtin/setup/system/latex/ubuntu.py,sha256=jAyzN-LloT9z6V2Ibm0uO-PaEvVu4Vq5c61LSvxq1Ww,584
|
79
|
-
zrb/builtin/setup/system/ubuntu.py,sha256=oOSN7Eq7arEpY2i0vWHPR2owio6dqqOvceteYrgmbYw,1019
|
75
|
+
zrb/builtin/setup/latex/ubuntu.py,sha256=er9wJAT4CpmghIaiIPFb3FvgqAn1aqU5UgX7GHL3FjA,577
|
76
|
+
zrb/builtin/setup/tmux/tmux.py,sha256=-MEIGP2OwM55Fi-ug1TjMgQGiTbwKlQ6aASSLLjjz7A,1630
|
77
|
+
zrb/builtin/setup/tmux/tmux_config.sh,sha256=wQCb4Q-mNkxIPOcvpN84X9RUWkGY16u3Vd-pOhVidgg,416
|
78
|
+
zrb/builtin/setup/tmux/tmux_helper.py,sha256=M03l0wfL25TzGGp6lnVfX40ayT_x7N2lz-nz2chO7PU,396
|
79
|
+
zrb/builtin/setup/ubuntu.py,sha256=oOSN7Eq7arEpY2i0vWHPR2owio6dqqOvceteYrgmbYw,1019
|
80
80
|
zrb/builtin/shell/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
81
81
|
zrb/builtin/shell/autocomplete/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
82
82
|
zrb/builtin/shell/autocomplete/bash.py,sha256=-7YDVV7txgJH9mAYSYN0jmvUEeDIzWFvVNY-cY0myF8,1181
|
83
83
|
zrb/builtin/shell/autocomplete/subcmd.py,sha256=WZI6cGWJcn80zSyxOHG7sCMO3Ucix3mZf4xm_xyB_Y0,606
|
84
84
|
zrb/builtin/shell/autocomplete/zsh.py,sha256=9hlq0Wt3fhRz326mAQTypEd4_4lZdrbBx_3A-Ti3mvw,1022
|
85
|
-
zrb/builtin/todo.py,sha256=
|
85
|
+
zrb/builtin/todo.py,sha256=2JTk0CQKGXyBll3l097CGRLEPCqAZ5f4luR7nL-Ad1o,9480
|
86
86
|
zrb/callback/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
87
87
|
zrb/callback/any_callback.py,sha256=Yhdv5UWHAZSVzj5K2JdxcVQx8x8VX8aZJEivj3NTfZc,247
|
88
88
|
zrb/callback/callback.py,sha256=IQ7r9EnXYHHcNXKBJAk4WFqCqj7WDvflAuCyu5y-27I,849
|
89
89
|
zrb/cmd/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
90
90
|
zrb/cmd/cmd_result.py,sha256=L8bQJzWCpcYexIxHBNsXj2pT3BtLmWex0iJSMkvimOA,597
|
91
91
|
zrb/cmd/cmd_val.py,sha256=LGuE85zQt0KSjpmW7bwIE5dnfIq-ULB8V2CDAiQ-SFk,960
|
92
|
-
zrb/config.py,sha256=
|
92
|
+
zrb/config.py,sha256=M__WVxBTU9zMxZvm1t2icEV8oU6-QK0Edl1XDyhJ9c8,3067
|
93
93
|
zrb/content_transformer/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
94
94
|
zrb/content_transformer/any_content_transformer.py,sha256=3XHM6ZdsJFXxRD7YlUkv0Gn7-mexsH8c8zdHt3C0x8k,741
|
95
95
|
zrb/content_transformer/content_transformer.py,sha256=8luyLZZneBIBAMgGZvydy0TTjhI-TWFgEiQbuCxCAbk,1640
|
@@ -110,17 +110,17 @@ zrb/group/any_group.py,sha256=1rNcsi5eu_86JAx_6Jy46SK4BTeppcb89MORynJd-4o,1115
|
|
110
110
|
zrb/group/group.py,sha256=KVaBZQ9Evihm84JCg5ZayWMbbA6iNTUiLmMgCk4OTcc,1862
|
111
111
|
zrb/input/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
112
112
|
zrb/input/any_input.py,sha256=07KgoPusxLMOwBHRjiImL955J33qC-24qGf_3My6fCE,701
|
113
|
-
zrb/input/base_input.py,sha256=
|
114
|
-
zrb/input/bool_input.py,sha256=
|
115
|
-
zrb/input/float_input.py,sha256=
|
116
|
-
zrb/input/int_input.py,sha256=
|
117
|
-
zrb/input/option_input.py,sha256=
|
118
|
-
zrb/input/password_input.py,sha256=
|
113
|
+
zrb/input/base_input.py,sha256=noLPZUZYJGhyNJ5b1cZyeOff6m7zUT-Cnh6n63S-DiM,2573
|
114
|
+
zrb/input/bool_input.py,sha256=m_mwzFtcAs26QIjVSiEDcqTQBukRtQ535RPGmH-b1_A,1384
|
115
|
+
zrb/input/float_input.py,sha256=FqpAIsszHw7-eFyOzVmXaXNcCCLEL1_M3XwsevsNWX0,1005
|
116
|
+
zrb/input/int_input.py,sha256=1knorHwOvSLuCPkcqA9iuYQG6Huy-8Vel5JNZqEFc44,1006
|
117
|
+
zrb/input/option_input.py,sha256=xG0K6uQCqEpsO0pRsOyPWPEE77PYl65t2VwtCfeHYbA,1941
|
118
|
+
zrb/input/password_input.py,sha256=C0h3kWJDI5KdXXQa-IXTQzyuCfjbx5tO34mn7kjmf2M,1313
|
119
119
|
zrb/input/str_input.py,sha256=NevZHX9rf1g8eMatPyy-kUX3DglrVAQpzvVpKAzf7bA,81
|
120
|
-
zrb/input/text_input.py,sha256=
|
120
|
+
zrb/input/text_input.py,sha256=ju-DgBRK3RpuWNomZXcw3V5QAgaMGeTFaNn_MKM13Xc,3055
|
121
121
|
zrb/runner/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
122
122
|
zrb/runner/cli.py,sha256=tM61thLwJNfjrHFVJqWmGuzE0MB273JZ8ANlROsKH1E,7024
|
123
|
-
zrb/runner/web_app.py,sha256=
|
123
|
+
zrb/runner/web_app.py,sha256=nO5EsGs8MMcvTdA41NU1yVlBu23HTD0FY82oXwOLIRM,6323
|
124
124
|
zrb/runner/web_controller/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
125
125
|
zrb/runner/web_controller/group_info_ui/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
126
126
|
zrb/runner/web_controller/group_info_ui/controller.py,sha256=sI1UKnRcr5K-a_lGMm5H54hbhdQwIMLlXYmwfxoIqSY,2831
|
@@ -192,16 +192,16 @@ zrb/util/git.py,sha256=QNGvJJ9kY1Hy6WNO4d22D_GXyC1l3yGtTOMJ9nXqqpc,5151
|
|
192
192
|
zrb/util/git_subtree.py,sha256=DidHZdKcpylomlJLiUyQJSpPSXaY399e-97H0WbOcvs,2619
|
193
193
|
zrb/util/group.py,sha256=Bg7HrSycoK110U5s_Tca6-uUQuZ5CMgb8wxZSrvDQ98,2790
|
194
194
|
zrb/util/llm/tool.py,sha256=9ezbPEAyMKLWABgeqAimJ82KBFl7ciWNFVGhtllXg6s,2106
|
195
|
-
zrb/util/load.py,sha256=
|
195
|
+
zrb/util/load.py,sha256=GEhvaS5ne25Pf5dPTOK_yuxZ-kHqZ0xe6WjfehLOAZA,1614
|
196
196
|
zrb/util/run.py,sha256=DGHUP9x1Q8V8UF3FbpmjLGuhVVCCLfjTH2teT8qXlNI,207
|
197
197
|
zrb/util/string/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
198
198
|
zrb/util/string/conversion.py,sha256=utfzaCLS2LTWMr9YbjXt2Z0a_pgzOjq76ZEPUh2UKCc,2876
|
199
199
|
zrb/util/string/format.py,sha256=c9dUrsl_DII7jZZPJi9NBTvxKnrzZDPcQuksW6FJ_Lk,611
|
200
200
|
zrb/util/string/name.py,sha256=8picJfUBXNpdh64GNaHv3om23QHhUZux7DguFLrXHp8,1163
|
201
|
-
zrb/util/todo.py,sha256=
|
201
|
+
zrb/util/todo.py,sha256=DEqBrEqzm2o9aATEyROtPsP4D7dbHIlTFVn4kvqOd4M,13498
|
202
202
|
zrb/xcom/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
203
203
|
zrb/xcom/xcom.py,sha256=P4aYHdE3FRsTsNrXGyW8N44IWZjw-vG_qys1Ymn3aBg,1572
|
204
|
-
zrb-1.0.
|
205
|
-
zrb-1.0.
|
206
|
-
zrb-1.0.
|
207
|
-
zrb-1.0.
|
204
|
+
zrb-1.0.0a12.dist-info/METADATA,sha256=yAtlpL0LBftuJRrC8zSn5erNKzCCkQAUGFFe6p_i9tg,4106
|
205
|
+
zrb-1.0.0a12.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
|
206
|
+
zrb-1.0.0a12.dist-info/entry_points.txt,sha256=-Pg3ElWPfnaSM-XvXqCxEAa-wfVI6BEgcs386s8C8v8,46
|
207
|
+
zrb-1.0.0a12.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|