forked from tldr-pages/tldr
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path_common.py
455 lines (336 loc) Β· 12 KB
/
_common.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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
#!/usr/bin/env python3
# SPDX-License-Identifier: MIT
"""
A Python file that makes some commonly used functions available for other scripts to use.
"""
from enum import Enum
from pathlib import Path
from unittest.mock import patch
import shutil
import os
import argparse
import subprocess
IGNORE_FILES = (".DS_Store",)
class Colors(str, Enum):
def __str__(self):
return str(
self.value
) # make str(Colors.COLOR) return the ANSI code instead of an Enum object
RED = "\x1b[31m"
GREEN = "\x1b[32m"
BLUE = "\x1b[34m"
CYAN = "\x1b[36m"
RESET = "\x1b[0m"
def test_ignore_files():
assert IGNORE_FILES == (".DS_Store",)
assert ".DS_Store" in IGNORE_FILES
assert "tldr.md" not in IGNORE_FILES
def get_tldr_root(lookup_path: Path = None) -> Path:
"""
Get the path of the local tldr repository, looking for it in each part of the given path. If it is not found, the path in the environment variable TLDR_ROOT is returned.
Parameters:
lookup_path (Path): the path to search for the tldr root. By default, the path of the script.
Returns:
Path: the local tldr repository.
"""
if lookup_path is None:
absolute_lookup_path = Path(__file__).resolve()
else:
absolute_lookup_path = Path(lookup_path).resolve()
if (
tldr_root := next(
(path for path in absolute_lookup_path.parents if path.name == "tldr"), None
)
) is not None:
return tldr_root
elif "TLDR_ROOT" in os.environ:
return Path(os.environ["TLDR_ROOT"])
raise SystemExit(
f"{Colors.RED}Please set the environment variable TLDR_ROOT to the location of a clone of https://github.com/tldr-pages/tldr{Colors.RESET}"
)
def test_get_tldr_root():
tldr_root = get_tldr_root("/path/to/tldr/scripts/test_script.py")
assert tldr_root == Path("/path/to/tldr")
# Set TLDR_ROOT in the environment
os.environ["TLDR_ROOT"] = "/path/to/tldr_clone"
tldr_root = get_tldr_root("/tmp")
assert tldr_root == Path("/path/to/tldr_clone")
del os.environ["TLDR_ROOT"]
# Remove TLDR_ROOT from the environment
original_env = os.environ.pop("TLDR_ROOT", None)
# Check if SystemExit is raised
raised = False
try:
get_tldr_root("/tmp")
except SystemExit:
raised = True
assert raised
# Restore the original values
if original_env is not None:
os.environ["TLDR_ROOT"] = original_env
def get_pages_dir(root: Path) -> list[Path]:
"""
Get all pages directories.
Parameters:
root (Path): the path to search for the pages directories.
Returns:
list (list of Path's): Path's of page entry and platform, e.g. "page.fr/common".
"""
return [d for d in root.iterdir() if d.name.startswith("pages")]
def test_get_pages_dir():
# Create temporary directories with names starting with "pages"
root = Path("test_root")
shutil.rmtree(root, True)
root.mkdir(exist_ok=True)
# Create temporary directories with names that do not start with "pages"
(root / "other_dir_1").mkdir(exist_ok=True)
(root / "other_dir_2").mkdir(exist_ok=True)
# Call the function and verify that it returns an empty list
result = get_pages_dir(root)
assert result == []
(root / "pages").mkdir(exist_ok=True)
(root / "pages.fr").mkdir(exist_ok=True)
(root / "other_dir").mkdir(exist_ok=True)
# Call the function and verify the result
result = get_pages_dir(root)
expected = [root / "pages", root / "pages.fr"]
assert result.sort() == expected.sort() # the order differs on Unix / macOS
shutil.rmtree(root, True)
def get_page_title(path: Path) -> str:
"""
Determine whether the given path has a title.
Parameters:
path (Path): Path to a page
Returns:
str: "" If the path doesn't exit or does not have a title,
otherwise return the page title.
"""
if not path.exists():
return ""
with path.open(encoding="utf-8") as f:
first_line = f.readline().strip()
return first_line.split("#", 1)[-1].strip()
def test_get_page_title():
# Test valid title
root = Path("test_root")
shutil.rmtree(root, True)
root.mkdir(exist_ok=True)
valid_path = root / "test.md"
valid_path.write_text("# Git Clone\nSome content", encoding="utf-8")
assert get_page_title(valid_path) == "Git Clone"
# Test title with multiple hashes
hash_path = root / "multiple_hash.md"
hash_path.write_text("# Git ### Clone\nSome content", encoding="utf-8")
assert get_page_title(hash_path) == "Git ### Clone"
# Test empty title
empty_path = root / "empty.md"
empty_path.write_text("#\nSome content", encoding="utf-8")
assert get_page_title(empty_path) == ""
# Test non-existent file
nonexistent_path = root / "nonexistent.md"
assert get_page_title(nonexistent_path) == ""
# Test title with leading/trailing spaces
spaces_path = root / "spaces.md"
spaces_path.write_text("# Git Clone \nSome content", encoding="utf-8")
assert get_page_title(spaces_path) == "Git Clone"
def get_target_paths(page: Path, pages_dirs: Path, language: str = "") -> list[Path]:
"""
Get all paths in all languages that match the page.
Parameters:
page (Path): the page to search for.
Returns:
list (list of Path's): A list of Path's.
"""
target_paths = []
if not page.lower().endswith(".md"):
page = f"{page}.md"
arg_platform, arg_page = page.split("/")
for pages_dir in pages_dirs:
if "." in pages_dir.name:
_, locale = pages_dir.name.split(".")
else:
locale = "en"
if language != "" and language != locale:
continue
page_path = pages_dir / arg_platform / arg_page
if not page_path.exists():
continue
target_paths.append(page_path)
target_paths.sort()
return target_paths
def test_get_target_paths():
root = Path("test_root")
shutil.rmtree(root, True)
root.mkdir(exist_ok=True)
shutil.os.makedirs(root / "pages" / "common")
shutil.os.makedirs(root / "pages.fr" / "common")
(root / "pages" / "common" / "tldr.md").touch()
(root / "pages.fr" / "common" / "tldr.md").touch()
target_paths = get_target_paths("common/tldr", get_pages_dir(root))
assert len(target_paths) == 2
assert all(p.name == "tldr.md" for p in target_paths)
fr_paths = get_target_paths("common/tldr", get_pages_dir(root), "fr")
assert len(fr_paths) == 1
assert str(fr_paths[0]).endswith("pages.fr/common/tldr.md")
en_paths = get_target_paths("common/tldr", get_pages_dir(root), "en")
assert len(en_paths) == 1
assert str(en_paths[0]).endswith("pages/common/tldr.md")
shutil.rmtree(root, True)
def get_locale(path: Path) -> str:
"""
Get the locale from the path.
Parameters:
path (Path): the path to extract the locale.
Returns:
str: a POSIX Locale Name in the form of "ll" or "ll_CC" (e.g. "fr" or "pt_BR").
"""
# compute locale
pages_dirname = path.parents[1].name
if "." in pages_dirname:
_, locale = pages_dirname.split(".")
else:
locale = "en"
return locale
def test_get_locale():
assert get_locale(Path("path/to/pages.fr/common/tldr.md")) == "fr"
assert get_locale(Path("path/to/pages/common/tldr.md")) == "en"
assert get_locale(Path("path/to/other/common/tldr.md")) == "en"
def get_status(action: str, dry_run: bool, type: str) -> str:
"""
Get a colored status line.
Parameters:
action (str): The action to perform.
dry_run (bool): Whether to perform a dry-run.
type (str): The kind of object to modify (alias, link).
Returns:
str: A colored line
"""
match action:
case "added":
start_color = Colors.CYAN
case "updated":
start_color = Colors.BLUE
case _:
start_color = Colors.RED
if dry_run:
status = f"{type} would be {action}"
else:
status = f"{type} {action}"
return create_colored_line(start_color, status)
def test_get_status():
# Test dry run status
assert (
get_status("added", True, "alias")
== f"{Colors.CYAN}alias would be added{Colors.RESET}"
)
assert (
get_status("updated", True, "link")
== f"{Colors.BLUE}link would be updated{Colors.RESET}"
)
# Test non-dry run status
assert (
get_status("added", False, "alias") == f"{Colors.CYAN}alias added{Colors.RESET}"
)
assert (
get_status("updated", False, "link")
== f"{Colors.BLUE}link updated{Colors.RESET}"
)
# Test default color for unknown action
assert (
get_status("unknown", True, "alias")
== f"{Colors.RED}alias would be unknown{Colors.RESET}"
)
def create_colored_line(start_color: str, text: str) -> str:
"""
Create a colored line.
Parameters:
start_color (str): The color for the line.
text (str): The text to display.
Returns:
str: A colored line
"""
return f"{start_color}{text}{Colors.RESET}"
def test_create_colored_line():
assert (
create_colored_line(Colors.CYAN, "TLDR") == f"{Colors.CYAN}TLDR{Colors.RESET}"
)
assert create_colored_line("Hello", "TLDR") == f"HelloTLDR{Colors.RESET}"
def create_argument_parser(description: str) -> argparse.ArgumentParser:
"""
Create an argument parser that can be extended.
Parameters:
description (str): The description for the argument parser
Returns:
ArgumentParser: an argument parser.
"""
parser = argparse.ArgumentParser(description=description)
parser.add_argument(
"-p",
"--page",
type=str,
default="",
help='page name in the format "platform/alias_command.md"',
)
parser.add_argument(
"-S",
"--sync",
action="store_true",
default=False,
help="synchronize each translation's alias page (if exists) with that of English page",
)
parser.add_argument(
"-l",
"--language",
type=str,
default="",
help='language in the format "ll" or "ll_CC" (e.g. "fr" or "pt_BR")',
)
parser.add_argument(
"-s",
"--stage",
action="store_true",
default=False,
help="stage modified pages (requires `git` to be on $PATH and TLDR_ROOT to be a Git repository)",
)
parser.add_argument(
"-n",
"--dry-run",
action="store_true",
default=False,
help="show what changes would be made without actually modifying the pages",
)
return parser
def test_create_argument_parser():
description = "Test argument parser"
parser = create_argument_parser(description)
assert isinstance(parser, argparse.ArgumentParser)
assert parser.description == description
# Check if each expected argument is added with the correct configurations
arguments = [
("-p", "--page", str, ""),
("-l", "--language", str, ""),
("-s", "--stage", None, False),
("-S", "--sync", None, False),
("-n", "--dry-run", None, False),
]
for short_flag, long_flag, arg_type, default_value in arguments:
action = parser._option_string_actions[short_flag] # Get action for short flag
assert action.dest.replace("_", "-") == long_flag.lstrip(
"-"
) # Check destination name
assert action.type == arg_type # Check argument type
assert action.default == default_value # Check default value
def stage(paths: list[Path]):
"""
Stage the given paths using Git.
Parameters:
paths (list of Paths): the list of Path's to stage using Git.
"""
subprocess.call(["git", "add", *(path.resolve() for path in paths)])
@patch("subprocess.call")
def test_stage(mock_subprocess_call):
paths = [Path("/path/to/file1"), Path("/path/to/file2")]
# Call the stage function
stage(paths)
# Verify that subprocess.call was called with the correct arguments
mock_subprocess_call.assert_called_once_with(["git", "add", *paths])