1
- #!/usr/bin/env -S deno run --allow-read --allow-run=bash,git,cargo --allow-env --allow-sys
1
+ #!/usr/bin/env -S deno run --allow-read --allow-run=bash,git,cargo --allow-net=docs.rs:443 --allow- env --allow-sys
2
2
3
3
import * as zx from "npm:zx"
4
- import { z , ZodSchema } from "https://deno.land/x/[email protected] /mod.ts"
5
- import { assertEquals } from "https://jsr.io/@std/assert/1.0.0/equals.ts"
6
- import { assert } from "https://jsr.io/@std/assert/1.0.0/assert.ts"
4
+ import { z , ZodSchema , ZodTypeDef } from "https://deno.land/x/[email protected] /mod.ts"
5
+ import { assert , assertEquals } from "jsr:@std/[email protected] "
7
6
8
- const CargoToml = z . object ( {
7
+ const CargoTomlSchema = z . object ( {
9
8
package : z . object ( {
10
9
name : z . string ( ) . min ( 1 ) ,
11
10
description : z . string ( ) . min ( 1 ) ,
12
11
repository : z . string ( ) . url ( ) . min ( 1 ) ,
13
12
metadata : z . object ( {
14
13
details : z . object ( {
15
- title : z . string ( ) . min ( 1 ) ,
16
- tagline : z . string ( ) ,
17
- summary : z . string ( ) ,
18
- } ) ,
19
- } ) ,
14
+ title : z . string ( ) . min ( 1 ) . optional ( ) ,
15
+ tagline : z . string ( ) . optional ( ) ,
16
+ summary : z . string ( ) . optional ( ) ,
17
+ peers : z . array ( z . string ( ) ) . default ( [ ] ) . describe ( "Packages that should be installed alongside this package" )
18
+ } ) . default ( { } ) ,
19
+ } ) . default ( { } ) ,
20
20
} ) ,
21
21
} )
22
22
23
- type CargoToml = z . infer < typeof CargoToml >
23
+ type CargoToml = z . infer < typeof CargoTomlSchema >
24
24
25
25
const CargoMetadataSchema = z . object ( {
26
26
packages : z . array ( z . object ( {
@@ -33,31 +33,70 @@ const CargoMetadataSchema = z.object({
33
33
34
34
type CargoMetadata = z . infer < typeof CargoMetadataSchema >
35
35
36
- const Repo = z . object ( {
36
+ const RepoSchema = z . object ( {
37
37
url : z . string ( ) . url ( ) ,
38
38
} )
39
39
40
- type Repo = z . infer < typeof Repo >
40
+ type Repo = z . infer < typeof RepoSchema >
41
41
42
- const $ = zx . $ ( {
43
- cwd : import . meta. dirname ,
42
+ const BadgeSchema = z . object ( {
43
+ name : z . string ( ) . min ( 1 ) ,
44
+ image : z . string ( ) . url ( ) ,
45
+ url : z . string ( ) . url ( ) ,
44
46
} )
45
47
46
- const parse = < T > ( schema : ZodSchema < T > , input : zx . ProcessOutput ) => schema . parse ( JSON . parse ( input . stdout ) )
47
- const renderMarkdownList = ( items : string [ ] ) => items . map ( ( bin ) => `* ${ bin } ` ) . join ( "\n" )
48
+ type Badge = z . infer < typeof BadgeSchema >
49
+
50
+ const badge = ( name : string , image : string , url : string ) : Badge => BadgeSchema . parse ( { name, url, image} )
51
+
52
+ const dirname = import . meta. dirname ;
53
+ if ( ! dirname ) throw new Error ( "Cannot determine the current script dirname" )
48
54
49
- const theCargoToml : CargoToml = parse ( CargoToml , await $ `yj -t < Cargo.toml` )
50
- const { package : { name, metadata : { details : { title } } } } = theCargoToml
55
+ const $ = zx . $ ( { cwd : dirname } )
56
+
57
+ // deno-lint-ignore no-explicit-any
58
+ const parse = < Output = any , Def extends ZodTypeDef = ZodTypeDef , Input = Output > ( schema : ZodSchema < Output , Def , Input > , input : zx . ProcessOutput ) => schema . parse ( JSON . parse ( input . stdout ) )
59
+
60
+ const theCargoToml : CargoToml = parse ( CargoTomlSchema , await $ `yj -t < Cargo.toml` )
61
+ const { package : { name, description, metadata : { details } } } = theCargoToml
62
+ const title = details . title || description
63
+ const peers = details . peers
51
64
const theCargoMetadata : CargoMetadata = parse ( CargoMetadataSchema , await $ `cargo metadata --format-version 1` )
52
65
const thePackageMetadata = theCargoMetadata . packages . find ( ( p ) => p . name == name )
53
66
assert ( thePackageMetadata , "Could not find package metadata" )
54
67
const target = thePackageMetadata . targets [ 0 ]
55
68
assert ( target , "Could not find package first target" )
56
- const doc = await $ `cargo doc2readme --template README.jl --target-name ${ target . name } --out -`
57
- const repo : Repo = parse ( Repo , await $ `gh repo view --json url` )
69
+ const docsUrl = `https://docs.rs/${ name } `
70
+
71
+ // launch multiple promises in parallel
72
+ const doc2ReadmePromise = $ `cargo doc2readme --template README.jl --target-name ${ target . name } --out -`
73
+ const ghRepoPromise = $ `gh repo view --json url`
74
+ const docsUrlPromise = fetch ( docsUrl , { method :"HEAD" } )
58
75
76
+ const doc = await doc2ReadmePromise
77
+ const docStr = doc . stdout . trim ( ) ;
78
+
79
+ const repo : Repo = parse ( RepoSchema , await ghRepoPromise )
59
80
assertEquals ( repo . url , theCargoToml . package . repository )
60
81
82
+ const docsUrlHead = await docsUrlPromise
83
+ const docsUrlIs200 = docsUrlHead . status === 200
84
+
85
+ const badges : Badge [ ] = [
86
+ badge ( "Build" , `${ repo . url } /actions/workflows/ci.yml/badge.svg` , repo . url )
87
+ ]
88
+ if ( docsUrlIs200 ) {
89
+ badges . push ( badge ( "Documentation" , `https://docs.rs/${ name } /badge.svg` , docsUrl ) )
90
+ }
91
+ const badgesStr = badges . map ( ( { name, image, url} ) => `[](${ url } )` ) . join ( "\n" ) ;
92
+
93
+ const titleSection = [
94
+ badgesStr ,
95
+ docStr
96
+ ] . filter ( s => s . length )
97
+
98
+ const cargoAddPackages = [ name , ...peers ] ;
99
+
61
100
const autogenerated = `
62
101
<!-- DO NOT EDIT -->
63
102
<!-- This file is automatically generated by README.ts. -->
@@ -69,15 +108,12 @@ ${autogenerated}
69
108
70
109
# ${ title }
71
110
72
- [](${ repo . url } )
73
- [](https://docs.rs/${ name } )
74
-
75
- ${ doc . stdout . trim ( ) }
111
+ ${ titleSection . join ( "\n\n" ) }
76
112
77
113
## Installation
78
114
79
115
\`\`\`shell
80
- cargo add ${ name }
116
+ cargo add ${ cargoAddPackages . join ( " " ) }
81
117
\`\`\`
82
118
83
119
## Gratitude
0 commit comments