Skip to content

Commit 85be985

Browse files
committed
Add links to the Function/Class signatures
add back the titles fix test fix test 2 fix anchor
1 parent 635efb6 commit 85be985

10 files changed

+211
-17
lines changed

scripts/js/lib/api/Pkg.test.ts

+68
Original file line numberDiff line numberDiff line change
@@ -113,3 +113,71 @@ test("Pkg.determineGithubUrlFn()", () => {
113113
"https://github.com/qiskit/qiskit-ibmq-provider/tree/stable/0.18/qiskit/providers/ibmq/foo.py",
114114
);
115115
});
116+
117+
test("Pkg.determineSignatureUrlFn()", () => {
118+
const runtime = Pkg.mock({
119+
name: "qiskit-ibm-runtime",
120+
type: "latest",
121+
versionWithoutPatch: "0.15",
122+
kebabCaseAndShortenUrls: true,
123+
}).determineSignatureUrlFn();
124+
125+
const qiskit = Pkg.mock({
126+
name: "qiskit",
127+
type: "latest",
128+
versionWithoutPatch: "0.45",
129+
kebabCaseAndShortenUrls: false,
130+
}).determineSignatureUrlFn();
131+
132+
const historicalQiskit = Pkg.mock({
133+
name: "qiskit",
134+
type: "historical",
135+
versionWithoutPatch: "0.32",
136+
kebabCaseAndShortenUrls: false,
137+
}).determineSignatureUrlFn();
138+
139+
const devQiskit = Pkg.mock({
140+
name: "qiskit",
141+
type: "dev",
142+
version: "1.0.0-dev",
143+
kebabCaseAndShortenUrls: false,
144+
}).determineSignatureUrlFn();
145+
146+
const qiskitC = Pkg.mock({
147+
name: "qiskit-c",
148+
type: "latest",
149+
versionWithoutPatch: "2.0.0",
150+
kebabCaseAndShortenUrls: true,
151+
}).determineSignatureUrlFn();
152+
153+
expect(runtime("../stubs/ibm_backend")).toEqual(
154+
"/api/qiskit-ibm-runtime/ibm-backend",
155+
);
156+
expect(runtime("apidoc/qiskit-ibm-runtime.ibm_backend")).toEqual(
157+
"/api/qiskit-ibm-runtime/ibm-backend",
158+
);
159+
160+
expect(qiskit("../stubs/exceptions")).toEqual("/api/qiskit/exceptions");
161+
expect(qiskit("apidocs/qasm2#anchor")).toEqual("/api/qiskit/qasm2#anchor");
162+
expect(qiskit("/apidoc/qasm3")).toEqual("/api/qiskit/qasm3");
163+
expect(qiskit("qiskit.transpiler.preset_passmanagers")).toEqual(
164+
"/api/qiskit/qiskit.transpiler.preset_passmanagers",
165+
);
166+
expect(qiskit("#test")).toEqual("#test");
167+
168+
expect(devQiskit("../stubs/exceptions")).toEqual(
169+
"/api/qiskit/dev/exceptions",
170+
);
171+
172+
expect(historicalQiskit("../apidocs/preset_passmanager")).toEqual(
173+
"/api/qiskit/0.32/preset_passmanager",
174+
);
175+
expect(
176+
historicalQiskit("stubs/qiskit.transpiler.preset_passmanagers"),
177+
).toEqual("/api/qiskit/0.32/qiskit.transpiler.preset_passmanagers");
178+
179+
expect(qiskitC("#test")).toEqual("#test");
180+
expect(qiskitC("cdoc/sparse_observable#test")).toEqual(
181+
"/api/qiskit-c/sparse-observable#test",
182+
);
183+
});

scripts/js/lib/api/Pkg.ts

+34
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,8 @@ import {
1919
QISKIT_TOC_GROUPING,
2020
QISKIT_ADDON_MPF_GROUPING,
2121
} from "./TocGrouping.js";
22+
import { removePart } from "../stringUtils.js";
23+
import { kebabCaseAndShortenPage } from "./normalizeResultUrls.js";
2224

2325
export class ReleaseNotesConfig {
2426
readonly enabled: boolean;
@@ -273,6 +275,38 @@ export class Pkg {
273275
return `${this.title}${versionStr} release notes`;
274276
}
275277

278+
determineSignatureUrlFn(): (rawLink: string) => string {
279+
let basePath = `/api/${this.name}`;
280+
if (this.isHistorical()) basePath += `/${this.versionWithoutPatch}`;
281+
if (this.isDev()) basePath += `/dev`;
282+
283+
return (rawLink: string) => {
284+
if (rawLink.startsWith("http") || rawLink.startsWith("#")) return rawLink;
285+
286+
const [rawHref, rawAnchor] = rawLink.split("#", 2);
287+
288+
const href = removePart(rawHref, "/", [
289+
"stubs",
290+
"apidocs",
291+
"apidoc",
292+
"cdoc",
293+
"..",
294+
]);
295+
296+
const segment = this.kebabCaseAndShortenUrls
297+
? kebabCaseAndShortenPage(href, this.name)
298+
: href;
299+
300+
const anchor = rawAnchor ? `#${rawAnchor}` : "";
301+
const link = `${segment}${anchor}`;
302+
303+
if (link.startsWith(basePath)) return link;
304+
305+
const separator = link.startsWith("/") ? "" : "/";
306+
return `${basePath}${separator}${link}`;
307+
};
308+
}
309+
276310
/**
277311
* Returns a function that takes in a fileName like `qiskit_ibm_runtime/job/exceptions` and returns the
278312
* stable GitHub URL to the file for this package's version.

scripts/js/lib/api/conversionPipeline.test.ts-snapshots/api-example.Electron

+1-1
Original file line numberDiff line numberDiff line change
@@ -102,7 +102,7 @@ python_api_name: api_example.Electron
102102

103103
### overloaded\_func
104104

105-
<Function id="api_example.Electron.overloaded_func" signature="overloaded_func(arg1: tuple[str, str], arg2: list[str], arg3: int, arg4: Electron) → None" extraSignatures={["overloaded_func(arg1: tuple[int, int], arg2: list[int], arg3: bool, arg4: set[Electron]) → None"]}>
105+
<Function id="api_example.Electron.overloaded_func" signature="overloaded_func(arg1: tuple[str, str], arg2: list[str], arg3: int, arg4: [Electron](#api_example.Electron)) → None" extraSignatures={["overloaded_func(arg1: tuple[int, int], arg2: list[int], arg3: bool, arg4: set[[Electron](#api_example.Electron)]) → None"]}>
106106
This is meant to test out [https://github.com/Qiskit/qiskit\_sphinx\_theme/pull/319](https://github.com/Qiskit/qiskit_sphinx_theme/pull/319).
107107

108108
**Parameters**

scripts/js/lib/api/conversionPipeline.ts

+1
Original file line numberDiff line numberDiff line change
@@ -107,6 +107,7 @@ async function convertFilesToMarkdown(
107107
html,
108108
fileName: file,
109109
determineGithubUrl: pkg.determineGithubUrlFn(),
110+
determineSignatureUrl: pkg.determineSignatureUrlFn(),
110111
imageDestination: pkg.outputDir("/images"),
111112
releaseNotesTitle: pkg.releaseNotesTitle(),
112113
hasSeparateReleaseNotes: pkg.hasSeparateReleaseNotes(),

scripts/js/lib/api/generateApiComponents.test.ts

+26-5
Original file line numberDiff line numberDiff line change
@@ -23,8 +23,13 @@ import { CheerioDoc } from "../testUtils.js";
2323

2424
const RAW_SIGNATURE_EXAMPLE = `<span class='sig-prename descclassname'><span class='pre'>Estimator.</span></span><span class='sig-name descname'><span class='pre'>run</span></span><span class='sig-paren'>(</span><em class='sig-param'><span class='n'><span class='pre'>circuits</span></span></em>, <em class='sig-param'><span class='n'><span class='pre'>observables</span></span></em>, <em class='sig-param'><span class='n'><span class='pre'>parameter_values</span></span><span class='o'><span class='pre'>=</span></span><span class='default_value'><span class='pre'>None</span></span></em>, <em class='sig-param'><span class='o'><span class='pre'>**</span></span><span class='n'><span class='pre'>kwargs</span></span></em><span class='sig-paren'>)</span></dt>`;
2525

26+
const determineSignatureUrl = (rawLink: string) => rawLink;
27+
2628
test("htmlSignatureToMd", async () => {
27-
const result = await htmlSignatureToMd(RAW_SIGNATURE_EXAMPLE);
29+
const result = await htmlSignatureToMd(
30+
RAW_SIGNATURE_EXAMPLE,
31+
determineSignatureUrl,
32+
);
2833
expect(result).toEqual(
2934
`Estimator.run(circuits, observables, parameter_values=None, **kwargs)`,
3035
);
@@ -71,7 +76,11 @@ test.describe("createOpeningTag()", () => {
7176
rawSignature: RAW_SIGNATURE_EXAMPLE,
7277
};
7378

74-
const tag = await createOpeningTag("Function", componentProps);
79+
const tag = await createOpeningTag(
80+
"Function",
81+
componentProps,
82+
determineSignatureUrl,
83+
);
7584
expect(tag).toEqual(`<Function
7685
id='qiskit_ibm_runtime.Estimator.run'
7786
attributeTypeHint='undefined'
@@ -92,7 +101,11 @@ test.describe("createOpeningTag()", () => {
92101
extraRawSignatures: [RAW_SIGNATURE_EXAMPLE, RAW_SIGNATURE_EXAMPLE],
93102
};
94103

95-
const tag = await createOpeningTag("Function", componentProps);
104+
const tag = await createOpeningTag(
105+
"Function",
106+
componentProps,
107+
determineSignatureUrl,
108+
);
96109
expect(tag).toEqual(`<Function
97110
id='qiskit_ibm_runtime.Estimator.run'
98111
attributeTypeHint='undefined'
@@ -113,7 +126,11 @@ test.describe("createOpeningTag()", () => {
113126
attributeValue: "None",
114127
};
115128

116-
const tag = await createOpeningTag("Attribute", componentProps);
129+
const tag = await createOpeningTag(
130+
"Attribute",
131+
componentProps,
132+
determineSignatureUrl,
133+
);
117134
expect(tag).toEqual(`<Attribute
118135
id='qiskit.circuit.QuantumCircuit.instance'
119136
attributeTypeHint='str | None'
@@ -132,7 +149,11 @@ test.describe("createOpeningTag()", () => {
132149
id: "qiskit.circuit.Sampler",
133150
};
134151

135-
const tag = await createOpeningTag("Class", componentProps);
152+
const tag = await createOpeningTag(
153+
"Class",
154+
componentProps,
155+
determineSignatureUrl,
156+
);
136157
expect(tag).toEqual(`<Class
137158
id='qiskit.circuit.Sampler'
138159
attributeTypeHint='undefined'

scripts/js/lib/api/generateApiComponents.ts

+45-3
Original file line numberDiff line numberDiff line change
@@ -15,13 +15,16 @@ import { unified } from "unified";
1515
import rehypeParse from "rehype-parse";
1616
import rehypeRemark from "rehype-remark";
1717
import remarkStringify from "remark-stringify";
18+
import { visit } from "unist-util-visit";
19+
import { toText } from "hast-util-to-text";
1820

1921
import { ApiType } from "./Metadata.js";
2022
import {
2123
getLastPartFromFullIdentifier,
2224
removeSuffix,
2325
APOSTROPHE_HEX_CODE,
2426
} from "../stringUtils.js";
27+
import { Root } from "mdast";
2528

2629
export type ComponentProps = {
2730
id?: string;
@@ -52,6 +55,7 @@ export async function processMdxComponent(
5255
priorApiType: ApiType | undefined,
5356
apiType: Exclude<ApiType, "module">,
5457
id: string,
58+
determineSignatureUrl: (rawLink: string) => string,
5559
): Promise<[string, string]> {
5660
const tagName = APITYPE_TO_TAG[apiType];
5761

@@ -71,7 +75,10 @@ export async function processMdxComponent(
7175
);
7276
addExtraSignatures(componentProps, extraProps);
7377

74-
return [await createOpeningTag(tagName, componentProps), `</${tagName}>`];
78+
return [
79+
await createOpeningTag(tagName, componentProps, determineSignatureUrl),
80+
`</${tagName}>`,
81+
];
7582
}
7683

7784
// ------------------------------------------------------------------
@@ -310,6 +317,7 @@ function prepareFunctionProps(
310317
export async function createOpeningTag(
311318
tagName: string,
312319
props: ComponentProps,
320+
determineSignatureUrl: (rawLink: string) => string,
313321
): Promise<string> {
314322
const attributeTypeHint = props.attributeTypeHint?.replaceAll(
315323
"'",
@@ -319,10 +327,15 @@ export async function createOpeningTag(
319327
"'",
320328
APOSTROPHE_HEX_CODE,
321329
);
322-
const signature = await htmlSignatureToMd(props.rawSignature!);
330+
const signature = await htmlSignatureToMd(
331+
props.rawSignature!,
332+
determineSignatureUrl,
333+
);
323334
const extraSignatures: string[] = [];
324335
for (const sig of props.extraRawSignatures ?? []) {
325-
extraSignatures.push(`"${await htmlSignatureToMd(sig!)}"`);
336+
extraSignatures.push(
337+
`"${await htmlSignatureToMd(sig!, determineSignatureUrl)}"`,
338+
);
326339
}
327340

328341
return `<${tagName}
@@ -398,14 +411,43 @@ export function addExtraSignatures(
398411
*/
399412
export async function htmlSignatureToMd(
400413
signatureHtml: string,
414+
determineSignatureUrl: (rawLink: string) => string,
401415
): Promise<string> {
402416
if (!signatureHtml) {
403417
return "";
404418
}
405419

420+
// The `code` tag helps us remove some undesired elements like asterisks surrounding
421+
// the parameters.
406422
const html = `<code>${signatureHtml}</code>`;
407423
const file = await unified()
408424
.use(rehypeParse)
425+
.use(() => (tree: Root) => {
426+
visit(tree, { tagName: "a" }, (node: any) => {
427+
// We transform the links into markdown as `text` nodes to avoid losing them once
428+
// transforming the signature from HTML to markdown. This could happen because we
429+
// have the signatures inside a `code` element. Moreover, we have more freedom by
430+
// manually creating the links.
431+
if (!node.properties?.href || !node.children?.length) {
432+
return;
433+
}
434+
435+
// We encode some conflicting characters that could make the markdown link break
436+
// by using URL-encoding form.
437+
const href = determineSignatureUrl(node.properties.href)
438+
.replaceAll('"', "%22")
439+
.replaceAll("(", "%28")
440+
.replaceAll(")", "%29");
441+
const isExternal = href.startsWith("http");
442+
const title = node.properties.title;
443+
444+
// We only show the title if it's from an external link
445+
const link = title && isExternal ? `${href} "${title}"` : `${href}`;
446+
447+
node.type = "text";
448+
node.value = `[${toText(node.children[0])}](${link})`;
449+
});
450+
})
409451
.use(rehypeRemark)
410452
.use(remarkStringify)
411453
.process(html);

scripts/js/lib/api/htmlToMd.test.ts

+1
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ const DEFAULT_ARGS = {
2020
`https://github.com/Qiskit/qiskit-ibm-runtime/tree/0.9.2/${fileName}.py`,
2121
releaseNotesTitle: "My Quantum release notes",
2222
hasSeparateReleaseNotes: false,
23+
determineSignatureUrl: (rawLink: string) => rawLink,
2324
};
2425

2526
async function toMd(

scripts/js/lib/api/htmlToMd.ts

+1
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@ export async function sphinxHtmlToMarkdown(options: {
3535
fileName: string;
3636
imageDestination: string;
3737
determineGithubUrl: (fileName: string) => string;
38+
determineSignatureUrl: (rawLink: string) => string;
3839
releaseNotesTitle: string;
3940
hasSeparateReleaseNotes: boolean;
4041
isCApi: boolean;

0 commit comments

Comments
 (0)