-
Notifications
You must be signed in to change notification settings - Fork 116
/
Copy pathupdate-relative-links.py
169 lines (148 loc) · 7.37 KB
/
update-relative-links.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
# Configure this script by setting the path (to files to update links for) and search_directory below.
# Path can either be a file or a folder. If given a folder, it will fix all relative links for .mdx files within the folder.
# Search_directory determines where the link fixer will search for similarly named files. Set this to the highest-up folder in which all other files exist.
# This script only fixes links between .md and .mdx files. Image paths are not checked.
# Use `python3 update-relative-links.py` to run this script.
import os
import re
def find_links(file_path):
"""
Find all relative markdown, href links in HTML/JSX, and href links in JSON in the provided file.
"""
with open(file_path, 'r') as file:
content = file.read()
# Regex to find markdown links: [text](relative/path.md)
markdown_links = re.findall(r'\[.*?\]\((.*?)\)', content)
# Regex to find href links: href="relative/path"
href_links_html = re.findall(r'href="(.*?)"', content)
# Regex to find href links in JSON: href: `relative/path` and href={`relative/path`}
href_links_json = re.findall(r'href:\s*`([^`]*)`|href=\{`([^`]*)`\}', content)
# Flatten the list of tuples returned by the regex for JSON links
href_links_json = [item for sublist in href_links_json for item in sublist if item]
return markdown_links + href_links_html + href_links_json, content
def search_directory_for_file(directory, filename):
"""
Recursively search the given directory and all subdirectories for a file with the given filename.
"""
for root, _, files in os.walk(directory):
if filename in files:
return os.path.relpath(os.path.join(root, filename), start=directory)
return None
def is_valid_link(link):
"""
Check if the link has a valid extension (.md, .mdx, or no extension).
"""
if link.startswith(('http://', 'https://', '#')):
return False
ext = os.path.splitext(link)[1]
if '#' in ext:
ext = ext.split('#')[0]
return ext in ('', '.md', '.mdx')
def update_links_in_file(file_path, directory, is_tsx_file=False):
"""
Update all relative markdown, href links in HTML/JSX, and href links in JSON in the file
with their actual paths found in the directory.
"""
links, content = find_links(file_path)
updated_content = content
warnings = False
changed = False
warning_messages = []
for link in links:
# Skip invalid links
if not is_valid_link(link):
continue
header = ''
if '#' in link:
filename, header = link.split('#', 1)
filename = os.path.basename(filename)
filename_mdx = filename if filename.endswith('.mdx') else f"{os.path.splitext(filename)[0]}.mdx"
else:
filename = os.path.basename(link)
if filename == "index":
filename_mdx = os.path.basename(os.path.dirname(link)) + ".mdx"
else:
filename_mdx = filename if filename.endswith('.mdx') else f"{os.path.splitext(filename)[0]}.mdx"
actual_path_md = search_directory_for_file(directory, filename)
actual_path_mdx = search_directory_for_file(directory, filename_mdx)
actual_path = actual_path_md if actual_path_md else actual_path_mdx
if actual_path:
# Calculate the relative path from the file to the actual path
relative_path = os.path.relpath(os.path.join(directory, actual_path), start=os.path.dirname(file_path))
# Replace en with ${locale} in the replacement relative link if in a .tsx file
if is_tsx_file:
split = relative_path.split('/en/')
if len(split) > 1:
relative_path = '/${locale}/' + split[1]
# Reattach the header if it exists
updated_link = f"{relative_path}#{header}" if header else relative_path
# Replace the relative link in the content
if f']({link})' in updated_content:
updated_content = updated_content.replace(f']({link})', f']({updated_link})')
changed = True
elif f'href="{link}"' in updated_content:
href_link = updated_link.rsplit('.', 1)[0] # Remove the .mdx extension
if link.startswith('./'):
href_link = './' + href_link
# Reattach the header if it exists for href links
if '#' in link:
href_link = f"{href_link}#{header}"
updated_content = updated_content.replace(f'href="{link}"', f'href="{href_link}"')
changed = True
elif f'href: `{link}`' in updated_content or f'href={{`{link}`}}' in updated_content:
href_link = '/' + relative_path.lstrip('/').rsplit('.', 1)[0] # Convert to absolute path without .mdx extension
split = href_link.split('/${locale}')
if len(split) > 1:
href_link = '/${locale}' + split[1]
# Reattach the header if it exists for JSON href links
if '#' in link:
href_link = f"{href_link}#{header}"
updated_content = updated_content.replace(f'href: `{link}`', f'href: `{href_link}`').replace(f'href={{`{link}`}}', f'href={{`{href_link}`}}')
changed = True
else:
# Output file and line number if unable to update link
line_number = content[:content.find(link)].count('\n') + 1
warning_messages.append(f"❌ Warning: Neither '{filename}' nor '{filename_mdx}' found.\n File: {file_path}#{line_number}")
warnings = True
with open(file_path, 'w') as file:
file.write(updated_content)
if not warnings:
if changed:
print(f"✅ Updated file: {file_path}\n")
else:
print(f"✅ Links are already up to date for: {file_path}\n")
else:
for message in warning_messages:
print(message)
print(f"Updated file with warnings: {file_path}")
print(f"Searched this directory: {directory}\n")
def update_links_in_folder(folder_path, search_directory):
"""
Recursively find all .mdx and .tsx files in the folder and update their links.
"""
for root, _, files in os.walk(folder_path):
for file in files:
if file.endswith(('.mdx', '.tsx')):
file_path = os.path.join(root, file)
is_tsx_file = file.endswith('.tsx')
update_links_in_file(file_path, search_directory, is_tsx_file)
def main(paths, search_directory):
"""
Main function to update links in the specified file or folder.
"""
if isinstance(paths, str):
paths = [paths]
for path in paths:
if not os.path.exists(path):
print(f"Error: Path '{path}' does not exist or is not a file/folder.")
continue
if os.path.isfile(path):
if path.endswith(('.mdx', '.tsx')):
is_tsx_file = path.endswith('.tsx')
update_links_in_file(path, search_directory, is_tsx_file)
elif os.path.isdir(path):
update_links_in_folder(path, search_directory)
# Update links across the docs site and in components. The landing page links are defined in components.
path = ["apps/nextra/pages/en", "apps/nextra/components"]
search_directory = "apps/nextra/pages/en"
main(path, search_directory)