@@ -3,6 +3,12 @@ import path from 'path';
3
3
import { execSync } from 'child_process' ;
4
4
import yaml from 'js-yaml' ;
5
5
6
+ /**
7
+ * Clones a Git repository to a specified directory
8
+ * @param {string } url The URL of the Git repository
9
+ * @param {string } directory The target directory to clone the repository to
10
+ * @returns
11
+ */
6
12
const cloneRepository = ( url , directory ) => {
7
13
try {
8
14
// Clone the repository
@@ -14,6 +20,12 @@ const cloneRepository = (url, directory) => {
14
20
}
15
21
} ;
16
22
23
+ /**
24
+ * Gets the GitHub URL of a Git repository from the local filesystem
25
+ * @param {string } rootPath The root path of the Git repository
26
+ * @returns {string } The GitHub URL of the repository
27
+ * @returns {null } If an error occurs
28
+ */
17
29
const getGitHubUrl = ( rootPath ) => {
18
30
try {
19
31
// Get the GitHub repository URL
@@ -29,6 +41,12 @@ const getGitHubUrl = (rootPath) => {
29
41
}
30
42
} ;
31
43
44
+ /**
45
+ * Gets the current Git branch of a local repository
46
+ * @param {string } rootPath The root path of the Git repository
47
+ * @returns {string } The name of the current branch
48
+ * @returns {null } If an error occurs
49
+ */
32
50
const getCurrentBranch = ( rootPath ) => {
33
51
try {
34
52
// Get the current branch name
@@ -40,42 +58,95 @@ const getCurrentBranch = (rootPath) => {
40
58
}
41
59
} ;
42
60
43
- const getRepositoryRoot = ( rootPath ) => {
61
+ /**
62
+ * Gets the root folder of a Git repository (the one containing the .git folder)
63
+ * @param {string } aPath The path to a file or directory within the repository
64
+ * @returns {string } The root folder of the repository
65
+ * @returns {null } If an error occurs
66
+ */
67
+ const getRepositoryRoot = ( aPath ) => {
44
68
try {
45
69
// Get the root folder of the repository
46
- const repositoryRoot = execSync ( 'git rev-parse --show-toplevel' , { cwd : rootPath , encoding : 'utf-8' } ) . trim ( ) ;
70
+ const repositoryRoot = execSync ( 'git rev-parse --show-toplevel' , { cwd : aPath , encoding : 'utf-8' } ) . trim ( ) ;
47
71
return repositoryRoot ;
48
72
} catch ( error ) {
49
73
console . error ( `Error getting repository root: ${ error . message } ` ) ;
50
74
return null ;
51
75
}
52
76
} ;
53
77
54
- const searchPackages = ( directory , outputFilename , indexUrl ) => {
78
+ /**
79
+ * Constructs a GitHub URL for a specific directory in a Git repository.
80
+ * @param {string } baseUrl The base URL of the GitHub repository
81
+ * @param {string } branch The current branch of the repository
82
+ * @param {string } repositoryRoot The root folder of the repository
83
+ * @param {string } dirPath The path to the directory within the repository
84
+ * @returns {string } The GitHub URL for the directory
85
+ */
86
+ const constructGitHubUrl = ( baseUrl , branch , repositoryRoot , dirPath ) => {
87
+ const relativePath = path . relative ( repositoryRoot , dirPath ) ;
88
+ const normalizedPath = relativePath . replace ( / \\ / g, '/' ) ; // Normalize path separators for Windows
89
+
90
+ return `${ baseUrl } /tree/${ branch } /${ normalizedPath } ` ;
91
+ } ;
92
+
93
+ /**
94
+ * Extracts the description from a manifest.py file
95
+ * @param {string } filePath The path to the manifest.py file
96
+ * @returns {string } The description extracted from the file
97
+ * @returns {null } If an error occurs
98
+ */
99
+ function extractDescription ( filePath ) {
100
+ try {
101
+ const content = fs . readFileSync ( filePath , 'utf8' ) ;
102
+ const descriptionMatch = / d e s c r i p t i o n = " ( .* ?) " / . exec ( content ) ;
103
+
104
+ if ( descriptionMatch && descriptionMatch [ 1 ] ) {
105
+ return descriptionMatch [ 1 ] ;
106
+ }
107
+ } catch ( error ) {
108
+ console . error ( `Error reading ${ filePath } : ${ error . message } ` ) ;
109
+ return null ;
110
+ }
111
+ }
112
+
113
+ /**
114
+ * Generates a list of packages from directories containing a package.json file
115
+ * @param {string } directory The starting directory to search for packages
116
+ * @param {string } indexUrl The URL of the package index to be assigned to each package
117
+ * @param {RegExp } excludePattern A regular expression to exclude certain directories
118
+ * @returns {string } A YAML representation of the package list as a string
119
+ */
120
+ const generatePackageList = ( directory , indexUrl , excludePattern ) => {
55
121
const result = { packages : [ ] } ;
56
122
57
123
const repositoryRoot = getRepositoryRoot ( directory ) ;
124
+ console . log ( `Repository root: ${ repositoryRoot } from ${ directory } ` ) ;
58
125
const gitHubUrl = getGitHubUrl ( repositoryRoot ) ;
59
126
const currentBranch = getCurrentBranch ( repositoryRoot ) ;
60
127
61
128
if ( ! repositoryRoot || ! gitHubUrl || ! currentBranch ) {
62
129
return ;
63
130
}
64
131
65
- const search = ( dir , rootPath ) => {
132
+ const collectPackages = ( dir , rootPath ) => {
66
133
const files = fs . readdirSync ( dir ) ;
67
134
68
135
for ( const file of files ) {
69
136
const filePath = path . join ( dir , file ) ;
70
137
const isDirectory = fs . statSync ( filePath ) . isDirectory ( ) ;
71
138
72
139
if ( isDirectory ) {
73
- search ( filePath , rootPath ) ;
140
+ collectPackages ( filePath , rootPath ) ;
74
141
} else {
75
142
const isPackageJson = file === 'package.json' ;
76
143
const isManifestPy = file === 'manifest.py' ;
77
144
const packageName = path . basename ( dir ) ;
78
145
146
+ if ( excludePattern && excludePattern . test ( dir ) ) {
147
+ continue ; // Skip excluded packages
148
+ }
149
+
79
150
if ( packageName . startsWith ( "_" ) ) {
80
151
continue ; // Skip "private" packages
81
152
}
@@ -89,16 +160,8 @@ const searchPackages = (directory, outputFilename, indexUrl) => {
89
160
} ;
90
161
91
162
if ( isManifestPy ) {
92
- try {
93
- const content = fs . readFileSync ( filePath , 'utf8' ) ;
94
- const descriptionMatch = / d e s c r i p t i o n = " ( .* ?) " / . exec ( content ) ;
95
-
96
- if ( descriptionMatch && descriptionMatch [ 1 ] ) {
97
- packageInfo . description = descriptionMatch [ 1 ] ;
98
- }
99
- } catch ( error ) {
100
- console . error ( `Error reading ${ file } : ${ error . message } ` ) ;
101
- }
163
+ const description = extractDescription ( filePath , packageInfo , file ) ;
164
+ packageInfo . description = description ;
102
165
}
103
166
104
167
result . packages . push ( packageInfo ) ;
@@ -107,22 +170,8 @@ const searchPackages = (directory, outputFilename, indexUrl) => {
107
170
}
108
171
} ;
109
172
110
- const constructGitHubUrl = ( baseUrl , branch , repositoryRoot , dirPath ) => {
111
- const relativePath = path . relative ( repositoryRoot , dirPath ) ;
112
- const normalizedPath = relativePath . replace ( / \\ / g, '/' ) ; // Normalize path separators for Windows
113
-
114
- return `${ baseUrl } /tree/${ branch } /${ normalizedPath } ` ;
115
- } ;
116
-
117
- search ( directory , repositoryRoot ) ;
118
-
119
- try {
120
- const yamlData = yaml . dump ( result ) ;
121
- fs . writeFileSync ( outputFilename , `---\n${ yamlData } ` ) ;
122
- console . log ( `YAML file saved to ${ outputFilename } ` ) ;
123
- } catch ( error ) {
124
- console . error ( `Error writing YAML file: ${ error . message } ` ) ;
125
- }
173
+ collectPackages ( directory , repositoryRoot ) ;
174
+ return yaml . dump ( result ) ;
126
175
} ;
127
176
128
177
// Check if command line arguments are provided
@@ -142,6 +191,13 @@ if (process.argv.length < 3) {
142
191
const directory = "build/micropython-lib" ;
143
192
const indexUrl = "https://micropython.org/pi/v2" ;
144
193
const outputFilename = process . argv [ 2 ] ;
194
+ const excludePattern = / \/ u n i x - f f i \/ / ; // Skip Unix-specific packages
145
195
146
- searchPackages ( directory , outputFilename , indexUrl ) ;
196
+ const packageList = generatePackageList ( directory , indexUrl , excludePattern ) ;
197
+ try {
198
+ fs . writeFileSync ( outputFilename , `---\n${ packageList } ` ) ;
199
+ console . log ( `YAML file saved to ${ outputFilename } ` ) ;
200
+ } catch ( error ) {
201
+ console . error ( `Error writing YAML file: ${ error . message } ` ) ;
202
+ }
147
203
}
0 commit comments