docs: support detecting broken link cross-projects (#10483)

* docs: support detecting broken link cross-projects

* remove double separators
This commit is contained in:
Shahed Nasser
2024-12-06 19:54:46 +02:00
committed by GitHub
parent a76b533604
commit e7e36f39fb
28 changed files with 492 additions and 166 deletions
+1 -1
View File
@@ -832,7 +832,7 @@ If you click on the workflow, you'll view a reference of that workflow, includin
This is useful if you want to extend an API route and pass additional data or perform custom actions.
Refer to [this guide](https://docs.medusajs.com/learn/customization/extend-features/extend-create-product) to find an example of extending an API route.
Refer to [this guide](!docs!/learn/customization/extend-features/extend-create-product) to find an example of extending an API route.
<Feedback
event="survey_api-ref"
@@ -4,7 +4,7 @@ import DownloadFull from "@/components/DownloadFull"
<H3 className="!mt-0">Just Getting Started?</H3>
Check out the [Medusa v2 Documentation](https://docs.medusajs.com).
Check out the [Medusa v2 Documentation](!docs!).
<Space bottom={8} />
@@ -16,7 +16,7 @@ To use Medusa's JS SDK library, install the following packages in your project (
npm install @medusajs/js-sdk@latest @medusajs/types@latest
```
Learn more about the JS SDK in [this documentation](https://docs.medusajs.com/resources/js-sdk).
Learn more about the JS SDK in [this documentation](!resources!/js-sdk).
### Download Full Reference
+1 -1
View File
@@ -831,7 +831,7 @@ If you click on the workflow, you'll view a reference of that workflow, includin
This is useful if you want to extend an API route and pass additional data or perform custom actions.
Refer to [this guide](https://docs.medusajs.com/learn/customization/extend-features/extend-create-product) to find an example of extending an API route.
Refer to [this guide](!docs!/learn/customization/extend-features/extend-create-product) to find an example of extending an API route.
<Feedback
event="survey_api-ref"
+47
View File
@@ -2,6 +2,11 @@ import createMDX from "@next/mdx"
import bundleAnalyzer from "@next/bundle-analyzer"
import rehypeMdxCodeProps from "rehype-mdx-code-props"
import rehypeSlug from "rehype-slug"
import {
brokenLinkCheckerPlugin,
crossProjectLinksPlugin,
} from "remark-rehype-plugins"
import path from "path"
/** @type {import('next').NextConfig} */
const nextConfig = {
@@ -28,6 +33,48 @@ const nextConfig = {
const withMDX = createMDX({
options: {
rehypePlugins: [
[
brokenLinkCheckerPlugin,
{
crossProjects: {
docs: {
projectPath: path.resolve("..", "book"),
},
resources: {
projectPath: path.resolve("..", "resources"),
hasGeneratedSlugs: true,
},
ui: {
projectPath: path.resolve("..", "ui"),
contentPath: "src/content/docs",
},
},
},
],
[
crossProjectLinksPlugin,
{
baseUrl: process.env.NEXT_PUBLIC_BASE_URL,
projectUrls: {
docs: {
url: process.env.NEXT_PUBLIC_DOCS_URL,
path: "",
},
resources: {
url: process.env.NEXT_PUBLIC_RESOURCES_URL,
},
"user-guide": {
url: process.env.NEXT_PUBLIC_USER_GUIDE_URL,
},
ui: {
url: process.env.NEXT_PUBLIC_UI_URL,
},
},
useBaseUrl:
process.env.NODE_ENV === "production" ||
process.env.VERCEL_ENV === "production",
},
],
[
rehypeMdxCodeProps,
{
+1
View File
@@ -41,6 +41,7 @@
"react-transition-group": "^4.4.5",
"rehype-mdx-code-props": "^3.0.1",
"rehype-slug": "^6.0.0",
"remark-rehype-plugins": "*",
"slugify": "^1.6.6",
"swr": "^2.2.0",
"tailwind": "*",
@@ -288,6 +288,6 @@ You can now execute this workflow in a custom API route, scheduled job, or subsc
<Note title="Tip">
Find a full list of the registered resources in the Medusa container and their registration key in [this reference](!resources!/resources/medusa-container-resources). You can use these resources in your custom workflows.
Find a full list of the registered resources in the Medusa container and their registration key in [this reference](!resources!/medusa-container-resources). You can use these resources in your custom workflows.
</Note>
@@ -6,7 +6,7 @@ export const metadata = {
# {metadata.title}
In the previous chapters, you [defined a link](../define-link/page.mdx) between the [custom Brand Module](../../custom-features/module/page.mdx) and Medusa's [Product Module](!resources!/comerce-modules/product), then [extended the create-product flow](../extend-create-product/page.mdx) to link a product to a brand.
In the previous chapters, you [defined a link](../define-link/page.mdx) between the [custom Brand Module](../../custom-features/module/page.mdx) and Medusa's [Product Module](!resources!/commerce-modules/product), then [extended the create-product flow](../extend-create-product/page.mdx) to link a product to a brand.
In this chapter, you'll learn how to retrieve a product's brand (and vice-versa) in two ways: Using Medusa's existing API route, or in customizations, such as a custom API route.
+16 -1
View File
@@ -9,11 +9,27 @@ import {
crossProjectLinksPlugin,
} from "remark-rehype-plugins"
import { sidebar } from "./sidebar.mjs"
import path from "path"
const withMDX = mdx({
extension: /\.mdx?$/,
options: {
rehypePlugins: [
[
brokenLinkCheckerPlugin,
{
crossProjects: {
resources: {
projectPath: path.resolve("..", "resources"),
hasGeneratedSlugs: true,
},
ui: {
projectPath: path.resolve("..", "ui"),
contentPath: "src/content/docs",
},
},
},
],
[
crossProjectLinksPlugin,
{
@@ -37,7 +53,6 @@ const withMDX = mdx({
process.env.VERCEL_ENV === "production",
},
],
[brokenLinkCheckerPlugin],
[localLinksRehypePlugin],
[
rehypeMdxCodeProps,
@@ -11,7 +11,7 @@ export const metadata = {
The Medusa Admin has two types of forms:
1. Create forms, created using the [FocusModal UI component](!ui!/components/focus-modal).
2. Edit or update forms, created using the [Drawer UI component](!ui!/ui/components/drawer).
2. Edit or update forms, created using the [Drawer UI component](!ui!/components/drawer).
This guide explains how to create these two form types following the Medusa Admin's conventions.
@@ -66,7 +66,7 @@ remoteLink.create({
<Note title="Tip">
Learn more about the remote link in [this documentation](!docs!/advanced-development/module-links/remote-link).
Learn more about the remote link in [this documentation](!docs!/learn/advanced-development/module-links/remote-link).
</Note>
@@ -14,7 +14,7 @@ In this document, you'll learn how to calculate a product variant's price with t
You'll need the following resources for the taxes calculation:
1. [Query](!docs!/advanced-development/module-links/query) to retrieve the product's variants' prices for a context. Learn more about that in [this guide](../price/page.mdx).
1. [Query](!docs!/learn/advanced-development/module-links/query) to retrieve the product's variants' prices for a context. Learn more about that in [this guide](../price/page.mdx).
2. The Tax Module's main service to get the tax lines for each product.
```ts
@@ -121,7 +121,7 @@ To do that, you'll consume the [promotionsCreated](/references/medusa-workflows/
<Note title="Tip">
Learn more about workflow hooks in [this guide](!docs!/advanced-development/workflows/workflow-hooks).
Learn more about workflow hooks in [this guide](!docs!/learn/advanced-development/workflows/workflow-hooks).
</Note>
@@ -156,7 +156,7 @@ In the snippet above, you add a validation rule indicating that `custom_name` is
<Note title="Tip">
Learn more about additional data validation in [this guide](!docs!/advanced-development/api-routes/additional-data).
Learn more about additional data validation in [this guide](!docs!/learn/advanced-development/api-routes/additional-data).
</Note>
@@ -208,7 +208,7 @@ In the compensation function that undoes the step's actions in case of an error,
<Note title="Tip">
Learn more about compensation functions in [this guide](!docs!/advanced-development/workflows/compensation-function).
Learn more about compensation functions in [this guide](!docs!/learn/advanced-development/workflows/compensation-function).
</Note>
@@ -266,9 +266,9 @@ The workflow accepts as an input the created promotion and the `additional_data`
In the workflow, you:
1. Use the `transform` utility to get the value of `custom_name` based on whether it's set in `additional_data`. Learn more about why you can't use conditional operators in a workflow without using `transform` in [this guide](!docs!/advanced-development/workflows/conditions#why-if-conditions-arent-allowed-in-workflows).
1. Use the `transform` utility to get the value of `custom_name` based on whether it's set in `additional_data`. Learn more about why you can't use conditional operators in a workflow without using `transform` in [this guide](!docs!/learn/advanced-development/workflows/conditions#why-if-conditions-arent-allowed-in-workflows).
2. Create the `Custom` record using the `createCustomStep`.
3. Use the `when-then` utility to link the promotion to the `Custom` record if it was created. Learn more about why you can't use if-then conditions in a workflow without using `when-then` in [this guide](!docs!/advanced-development/workflows/conditions#why-if-conditions-arent-allowed-in-workflows).
3. Use the `when-then` utility to link the promotion to the `Custom` record if it was created. Learn more about why you can't use if-then conditions in a workflow without using `when-then` in [this guide](!docs!/learn/advanced-development/workflows/conditions#why-if-conditions-arent-allowed-in-workflows).
You'll next execute the workflow in the hook handler.
@@ -379,7 +379,7 @@ Among the returned `promotion` object, you'll find a `custom` property which hol
### Retrieve using Query
You can also retrieve the `Custom` record linked to a promotion in your code using [Query](!docs!/advanced-development/module-links/query).
You can also retrieve the `Custom` record linked to a promotion in your code using [Query](!docs!/learn/advanced-development/module-links/query).
For example:
@@ -393,7 +393,7 @@ const { data: [promotion] } = await query.graph({
})
```
Learn more about how to use Query in [this guide](!docs!/advanced-development/module-links/query).
Learn more about how to use Query in [this guide](!docs!/learn/advanced-development/module-links/query).
---
+1 -1
View File
@@ -736,7 +736,7 @@ The Medusa Admin plugin can be extended to add widgets, new pages, and setting p
icon: AcademicCapSolid,
},
{
href: "!docs!/learn/advanced-development/admin/setting-pages",
href: "!docs!/learn/advanced-development/admin/ui-routes#create-settings-page",
title: "Create Admin Setting Page",
text: "Learn how to add new page to the Medusa Admin settings.",
icon: AcademicCapSolid,
@@ -2008,7 +2008,7 @@ This loops over the returned subscriptions and executes the `createSubscriptionO
### Further Reads
- [How to Create a Scheduled Job](!docs!/learn/basics/scheeduled-jobs)
- [How to Create a Scheduled Job](!docs!/learn/basics/scheduled-jobs)
---
-3
View File
@@ -9118,9 +9118,6 @@ export const generatedSidebar = [
}
]
},
{
"type": "separator"
},
{
"loaded": true,
"isPathHref": true,
+15 -1
View File
@@ -7,13 +7,27 @@ import {
workflowDiagramLinkFixerPlugin,
} from "remark-rehype-plugins"
import mdxPluginOptions from "./mdx-options.mjs"
import path from "node:path"
const withMDX = mdx({
extension: /\.mdx?$/,
options: {
rehypePlugins: [
[
brokenLinkCheckerPlugin,
{
crossProjects: {
docs: {
projectPath: path.resolve("..", "book"),
},
ui: {
projectPath: path.resolve("..", "ui"),
contentPath: "src/content/docs",
},
},
},
],
...mdxPluginOptions.options.rehypePlugins,
[brokenLinkCheckerPlugin],
[localLinksRehypePlugin],
[typeListLinkFixerPlugin],
[
-3
View File
@@ -2131,9 +2131,6 @@ export const sidebar = sidebarAttachHrefCommonOptions([
},
],
},
{
type: "separator",
},
{
type: "category",
title: "General",
+2 -2
View File
@@ -7,7 +7,7 @@ const monoRepoPath = path.resolve("..", "..", "..")
/**
*
* @param {string} dir - The directory to search in
* @returns {Promise<{ origSlug: string; newSlug: string }[]>}
* @returns {Promise<import("types").SlugChange[]>}
*/
export default async function getSlugs(options = {}) {
let { dir, basePath = path.resolve("app"), baseSlug = basePath } = options
@@ -15,7 +15,7 @@ export default async function getSlugs(options = {}) {
dir = basePath
}
/**
* @type {{ origSlug: string; newSlug: string }[]}
* @type {import("types").SlugChange[]}
*/
const slugs = []
@@ -1,9 +1,253 @@
import { existsSync } from "fs"
import { existsSync, readdirSync, readFileSync } from "fs"
import path from "path"
import type { Transformer } from "unified"
import type { UnistNode, UnistTree } from "./types/index.js"
import type {
BrokenLinkCheckerOptions,
UnistNode,
UnistNodeWithData,
UnistTree,
} from "./types/index.js"
import type { VFile } from "vfile"
import { parseCrossProjectLink } from "./utils/cross-project-link-utils.js"
import { SlugChange } from "types"
import getAttribute from "./utils/get-attribute.js"
import { estreeToJs } from "./utils/estree-to-js.js"
import { performActionOnLiteral } from "./utils/perform-action-on-literal.js"
import { MD_LINK_REGEX } from "./constants.js"
export function brokenLinkCheckerPlugin(): Transformer {
function getErrorMessage({
link,
file,
}: {
link: string
file: VFile
}): string {
return `Broken link found! ${link} linked in ${file.history[0]}`
}
function checkLocalLinkExists({
link,
file,
currentPageFilePath,
}: {
link: string
file: VFile
currentPageFilePath: string
}) {
// get absolute path of the URL
const linkedFilePath = path
.resolve(currentPageFilePath, link)
.replace(/#.*$/, "")
// check if the file exists
if (!existsSync(linkedFilePath)) {
throw new Error(
getErrorMessage({
link,
file,
})
)
}
}
function mdxPageExists(pagePath: string): boolean {
if (!existsSync(pagePath)) {
// for projects that use a convention other than mdx
// check if an mdx file exists with the same name
if (existsSync(`${pagePath}.mdx`)) {
return true
}
return false
}
if (existsSync(path.join(pagePath, "page.mdx"))) {
return true
}
// for projects that use a convention other than mdx
// check if an mdx file exists with the same name
return readdirSync(pagePath).some((fileName) => fileName.endsWith(".mdx"))
}
function componentChecker({
node,
...rest
}: {
node: UnistNodeWithData
file: VFile
currentPageFilePath: string
options: BrokenLinkCheckerOptions
}) {
if (!node.name) {
return
}
let attributeName: string | undefined
const maybeCheckAttribute = () => {
if (!attributeName) {
return
}
const attribute = getAttribute(node, attributeName)
if (
!attribute ||
typeof attribute.value === "string" ||
!attribute.value.data?.estree
) {
return
}
const itemJsVar = estreeToJs(attribute.value.data.estree)
if (!itemJsVar) {
return
}
performActionOnLiteral(itemJsVar, (item) => {
checkLink({
link: item.original.value as string,
...rest,
})
})
}
switch (node.name) {
case "Prerequisites":
case "CardList":
attributeName = "items"
break
case "Card":
attributeName = "href"
break
case "WorkflowDiagram":
attributeName = "workflow"
break
case "TypeList":
attributeName = "types"
break
}
maybeCheckAttribute()
}
function checkLink({
link,
file,
currentPageFilePath,
options,
}: {
link: unknown | undefined
file: VFile
currentPageFilePath: string
options: BrokenLinkCheckerOptions
}) {
if (!link || typeof link !== "string") {
return
}
// try to remove hash
const hashIndex = link.lastIndexOf("#")
const likeWithoutHash = hashIndex !== -1 ? link.substring(0, hashIndex) : link
if (likeWithoutHash.match(/page\.mdx?$/)) {
checkLocalLinkExists({
link: likeWithoutHash,
file,
currentPageFilePath,
})
return
}
const parsedLink = parseCrossProjectLink(likeWithoutHash)
if (!parsedLink || !Object.hasOwn(options.crossProjects, parsedLink.area)) {
if (MD_LINK_REGEX.test(link)) {
// try fixing MDX links
let linkMatches
let tempLink = link
MD_LINK_REGEX.lastIndex = 0
while ((linkMatches = MD_LINK_REGEX.exec(tempLink)) !== null) {
if (!linkMatches.groups?.link) {
return
}
checkLink({
link: linkMatches.groups.link,
file,
currentPageFilePath,
options,
})
tempLink = tempLink.replace(linkMatches.groups.link, "")
// reset regex
MD_LINK_REGEX.lastIndex = 0
}
}
return
}
const projectOptions = options.crossProjects[parsedLink.area]
const isReferenceLink = parsedLink.path.startsWith("/references")
const baseDir = isReferenceLink
? "references"
: projectOptions.contentPath || "app"
const pagePath = isReferenceLink
? parsedLink.path.replace(/^\/references/, "")
: parsedLink.path
// check if the file exists
if (mdxPageExists(path.join(projectOptions.projectPath, baseDir, pagePath))) {
return
}
// file doesn't exist, check if slugs are enabled and generated
const generatedSlugsPath = path.join(
projectOptions.projectPath,
"generated",
"slug-changes.mjs"
)
if (!projectOptions.hasGeneratedSlugs || !existsSync(generatedSlugsPath)) {
throw new Error(
getErrorMessage({
link,
file,
})
)
}
// get slugs from file
const generatedSlugContent = readFileSync(generatedSlugsPath, "utf-8")
const slugChanges: SlugChange[] = JSON.parse(
generatedSlugContent.substring(generatedSlugContent.indexOf("["))
)
const slugChange = slugChanges.find(
(change) => change.newSlug === parsedLink.path
)
if (
!slugChange ||
!mdxPageExists(path.join(projectOptions.projectPath, slugChange.origSlug))
) {
throw new Error(
getErrorMessage({
link,
file,
})
)
}
}
const allowedComponentNames = [
"Card",
"CardList",
"Prerequisites",
"WorkflowDiagram",
"TypeList",
]
export function brokenLinkCheckerPlugin(
options: BrokenLinkCheckerOptions
): Transformer {
return async (tree, file) => {
const { visit } = await import("unist-util-visit")
@@ -12,20 +256,26 @@ export function brokenLinkCheckerPlugin(): Transformer {
""
)
visit(tree as UnistTree, "element", (node: UnistNode) => {
if (node.tagName !== "a" || !node.properties?.href?.match(/page\.mdx?/)) {
return
visit(
tree as UnistTree,
["element", "mdxJsxFlowElement"],
(node: UnistNode) => {
if (node.tagName === "a" && node.properties?.href) {
checkLink({
link: node.properties.href,
file,
currentPageFilePath,
options,
})
} else if (node.name && allowedComponentNames.includes(node.name)) {
componentChecker({
node: node as UnistNodeWithData,
file,
currentPageFilePath,
options,
})
}
}
// get absolute path of the URL
const linkedFilePath = path
.resolve(currentPageFilePath, node.properties.href)
.replace(/#.*$/, "")
// check if the file exists
if (!existsSync(linkedFilePath)) {
throw new Error(
`Broken link found! ${node.properties.href} linked in ${file.history[0]}`
)
}
})
)
}
}
@@ -0,0 +1 @@
export const MD_LINK_REGEX = /\[(.*?)\]\((?<link>(![a-z]+!|\.).*?)\)/gm
@@ -1,18 +1,13 @@
/* eslint-disable no-case-declarations */
import type { Transformer } from "unified"
import type {
CrossProjectLinksOptions,
ExpressionJsVar,
UnistNode,
UnistNodeWithData,
UnistTree,
} from "./types/index.js"
import { estreeToJs } from "./utils/estree-to-js.js"
import getAttribute from "./utils/get-attribute.js"
import {
isExpressionJsVarLiteral,
isExpressionJsVarObj,
} from "./utils/expression-is-utils.js"
import { performActionOnLiteral } from "./utils/perform-action-on-literal.js"
const PROJECT_REGEX = /^!(?<area>[\w-]+)!/
@@ -61,89 +56,65 @@ function componentFixer(
return
}
const fixProperty = (item: ExpressionJsVar) => {
if (!isExpressionJsVarObj(item)) {
let attributeName: string | undefined
const maybeCheckAttribute = () => {
if (!attributeName) {
return
}
Object.entries(item).forEach(([key, value]) => {
if (
(key !== "href" && key !== "link") ||
!isExpressionJsVarLiteral(value)
) {
return
}
const attribute = getAttribute(node, attributeName)
value.original.value = matchAndFixLinks(
value.original.value as string,
if (
!attribute ||
typeof attribute.value === "string" ||
!attribute.value.data?.estree
) {
return
}
const itemJsVar = estreeToJs(attribute.value.data.estree)
if (!itemJsVar) {
return
}
performActionOnLiteral(itemJsVar, (item) => {
item.original.value = matchAndFixLinks(
item.original.value as string,
options
)
value.original.raw = JSON.stringify(value.original.value)
item.original.raw = JSON.stringify(item.original.value)
})
}
switch (node.name) {
case "CardList":
const itemsAttribute = getAttribute(node, "items")
if (
!itemsAttribute?.value ||
typeof itemsAttribute.value === "string" ||
!itemsAttribute.value.data?.estree
) {
return
}
const jsVar = estreeToJs(itemsAttribute.value.data.estree)
if (!jsVar) {
return
}
if (Array.isArray(jsVar)) {
jsVar.forEach(fixProperty)
} else {
fixProperty(jsVar)
}
return
case "Card":
const hrefAttribute = getAttribute(node, "href")
if (!hrefAttribute?.value || typeof hrefAttribute.value !== "string") {
return
}
hrefAttribute.value = matchAndFixLinks(hrefAttribute.value, options)
return
case "Prerequisites":
const prerequisitesItemsAttribute = getAttribute(node, "items")
if (
!prerequisitesItemsAttribute?.value ||
typeof prerequisitesItemsAttribute.value === "string" ||
!prerequisitesItemsAttribute.value.data?.estree
) {
return
}
const prerequisitesJsVar = estreeToJs(
prerequisitesItemsAttribute.value.data.estree
)
if (!prerequisitesJsVar) {
return
}
if (Array.isArray(prerequisitesJsVar)) {
prerequisitesJsVar.forEach(fixProperty)
} else {
fixProperty(prerequisitesJsVar)
}
return
attributeName = "items"
break
case "Card":
attributeName = "href"
break
case "WorkflowDiagram":
attributeName = "workflow"
break
case "TypeList":
attributeName = "types"
break
}
maybeCheckAttribute()
}
const allowedComponentNames = [
"Card",
"CardList",
"Prerequisites",
"WorkflowDiagram",
"TypeList",
]
export function crossProjectLinksPlugin(
options: CrossProjectLinksOptions
): Transformer {
@@ -155,9 +126,7 @@ export function crossProjectLinksPlugin(
["element", "mdxJsxFlowElement"],
(node: UnistNode) => {
const isComponent =
node.name === "Card" ||
node.name === "CardList" ||
node.name === "Prerequisites"
node.name && allowedComponentNames.includes(node.name)
const isLink = node.tagName === "a" && node.properties?.href
if (!isComponent && !isLink) {
return
@@ -118,6 +118,16 @@ export declare type CrossProjectLinksOptions = {
useBaseUrl?: boolean
}
export declare type BrokenLinkCheckerOptions = {
crossProjects: {
[k: string]: {
projectPath: string
contentPath?: string
hasGeneratedSlugs?: boolean
}
}
}
export declare type ComponentLinkFixerLinkType = "md" | "value"
export declare type ComponentLinkFixerOptions = {
@@ -1,21 +1,13 @@
import path from "path"
import { Transformer } from "unified"
import {
ComponentLinkFixerLinkType,
ExpressionJsVar,
UnistNodeWithData,
UnistTree,
} from "../types/index.js"
import { UnistNodeWithData, UnistTree } from "../types/index.js"
import { FixLinkOptions, fixLinkUtil } from "../index.js"
import getAttribute from "../utils/get-attribute.js"
import { estreeToJs } from "../utils/estree-to-js.js"
import {
isExpressionJsVarLiteral,
isExpressionJsVarObj,
} from "../utils/expression-is-utils.js"
import { ComponentLinkFixerOptions } from "../types/index.js"
import { performActionOnLiteral } from "./perform-action-on-literal.js"
import { MD_LINK_REGEX } from "../constants.js"
const MD_LINK_REGEX = /\[(.*?)\]\((?<link>(![a-z]+!|\.).*?)\)/gm
const VALUE_LINK_REGEX = /^(![a-z]+!|\.)/gm
function matchMdLinks(
@@ -59,33 +51,6 @@ function matchValueLink(
})
}
function traverseJsVar(
item: ExpressionJsVar[] | ExpressionJsVar,
linkOptions: Omit<FixLinkOptions, "linkedPath">,
checkLinksType: ComponentLinkFixerLinkType
) {
const linkFn = checkLinksType === "md" ? matchMdLinks : matchValueLink
if (Array.isArray(item)) {
item.forEach((item) => traverseJsVar(item, linkOptions, checkLinksType))
} else if (isExpressionJsVarLiteral(item)) {
item.original.value = linkFn(item.original.value as string, linkOptions)
item.original.raw = JSON.stringify(item.original.value)
} else {
Object.values(item).forEach((value) => {
if (Array.isArray(value) || isExpressionJsVarObj(value)) {
return traverseJsVar(value, linkOptions, checkLinksType)
}
if (!isExpressionJsVarLiteral(value)) {
return
}
value.original.value = linkFn(value.original.value as string, linkOptions)
value.original.raw = JSON.stringify(value.original.value)
})
}
}
export function componentLinkFixer(
componentName: string,
attributeName: string,
@@ -117,12 +82,12 @@ export function componentLinkFixer(
return
}
const workflowAttribute = getAttribute(node, attributeName)
const attribute = getAttribute(node, attributeName)
if (
!workflowAttribute ||
typeof workflowAttribute.value === "string" ||
!workflowAttribute.value.data?.estree
!attribute ||
typeof attribute.value === "string" ||
!attribute.value.data?.estree
) {
return
}
@@ -132,13 +97,17 @@ export function componentLinkFixer(
appsPath,
}
const itemJsVar = estreeToJs(workflowAttribute.value.data.estree)
const itemJsVar = estreeToJs(attribute.value.data.estree)
if (!itemJsVar) {
return
}
traverseJsVar(itemJsVar, linkOptions, checkLinksType)
const linkFn = checkLinksType === "md" ? matchMdLinks : matchValueLink
performActionOnLiteral(itemJsVar, (item) => {
item.original.value = linkFn(item.original.value as string, linkOptions)
item.original.raw = JSON.stringify(item.original.value)
})
})
}
}
@@ -0,0 +1,21 @@
const PROJECT_REGEX = /^!(?<area>[\w-]+)!/
export const parseCrossProjectLink = (
link: string
):
| {
area: string
path: string
}
| undefined => {
const projectArea = PROJECT_REGEX.exec(link)
if (!projectArea?.groups?.area) {
return undefined
}
return {
area: projectArea.groups.area,
path: link.replace(PROJECT_REGEX, ""),
}
}
@@ -0,0 +1,28 @@
import { ExpressionJsVar, ExpressionJsVarLiteral } from "../types/index.js"
import {
isExpressionJsVarLiteral,
isExpressionJsVarObj,
} from "./expression-is-utils.js"
export const performActionOnLiteral = (
item: ExpressionJsVar[] | ExpressionJsVar,
action: (item: ExpressionJsVarLiteral) => void
) => {
if (Array.isArray(item)) {
item.forEach((i) => performActionOnLiteral(i, action))
} else if (isExpressionJsVarLiteral(item)) {
action(item)
} else {
Object.values(item).forEach((value) => {
if (Array.isArray(value) || isExpressionJsVarObj(value)) {
return performActionOnLiteral(value, action)
}
if (!isExpressionJsVarLiteral(value)) {
return
}
action(value)
})
}
}
+5
View File
@@ -0,0 +1,5 @@
export type SlugChange = {
origSlug: string
newSlug: string
filePath: string
}
+1
View File
@@ -1,4 +1,5 @@
export * from "./api-testing.js"
export * from "./build-scripts.js"
export * from "./config.js"
export * from "./general.js"
export * from "./menu.js"
+1
View File
@@ -6805,6 +6805,7 @@ __metadata:
react-transition-group: ^4.4.5
rehype-mdx-code-props: ^3.0.1
rehype-slug: ^6.0.0
remark-rehype-plugins: "*"
slugify: ^1.6.6
swr: ^2.2.0
tailwind: "*"