Description
Bug Report
Docsify's default path handling is problematic and/or broken in several scenarios:
-
Docsify's path behavior is inconsistent
Docsify handles paths differently for links and other resources like images and embedded content. This inconsistency requires users to research and memorize how and why path behavior varies based on resource type which is unnecessarily confusing and makes Docsify unintuitive for users. -
Docsify's relative path behavior is non-standard
Relative paths are generally expected to be relative to the context in which they are used. In the context of a website made up of a collection of separate pages, this means relative paths should be relative to the page where they are used. This provides document portability by allowing links between documents and resources to remain intact as they are moved throughout the hierarchy so long as their location relative to each other remains the same.- 🟢 Docsify treats relative non-link paths (images, embedded content, ...) as relative to the current page route. This is the correct behavior for relative paths.
- 🟡 Docsify treats relative link paths as relative to a site's
index.html
route. This makes sense from a technical perspective, but this is not expected behavior based on how relative paths work elsewhere. For sites with a flat hierarchy where theindex.html
and markdown pages are located at the root (/
) this isn't an issue, but for sites with nested routes (e.g.,/dir/page.md
) or a nestedindex.html
(e.g.,/docs/index.html
) this results in confusion, the need to use unintuitive paths, and broken or incorrect links.
-
Docsify's absolute path behavior is non-standard
Absolute paths are generally expected to be relative to the top-most level in a domain or directory hierarchy. In the context of a Docsify site, this most commonly means the root of the domain (/
), but nested sites (/docs/index.html
) may prefer to have absolute paths relative to theindex.html
route for site portability. This provides consistent and reliable paths to resources from any level within the hierarchy (e.g., accessing/assets/image.png
from/
,/foo/
, or/foo/bar/baz
).- 🟡 Docsify treats absolute non-link paths (images, embedded content, ...) as relative to the current page route. This is not how absolute paths work elsewhere which causes confusion, the need to use unintuitive paths, and broken or incorrect links.
- 🟡 Docsify treats relative link paths as relative to a site's
index.html
route. This is not how absolute paths work elsewhere which causes confusion, the need to use unintuitive paths, and broken or incorrect links. - 🔴 Docsify's handling of absolute paths prevents absolute paths from being used to access resources above the site's level in the hierarchy. For example, it is not possible to access
/assets/image.png
from/dir/page.md
using an absolute URL.
-
Docsify's absolute path behavior is broken for nested routes
🔴 In Docsify attempt to treat absolute non-link paths (images, embedded content, ...) as relative to the current page route, it incorrectly adds the current page route to paths that already contain the page route. For example:[Root] └─ docs ├─ dir │ ├─ image.png │ └─ page.md ├─ index.html └─ README.md
<!-- /docs/dir/page.md --> 
Result:
<img src="/dir/dir/image.png">
-
Docsify does not provide a way to modify or opt-out of its problematic handling of absolute paths
This has led users resorting to using HTML instead of markdown and PRs like fix: render image in absolute path #1868 which propose working around these issues by introducing more options or custom markdown syntax. None of these would be necessary if paths simply worked the way people expected them to by default. Docsify does offer configuration options likebasePath
andrelativePath
, but neither of these issues allows paths in Docsify to work the way they do outside of Docsify and as people expect them to.
The result has been many issues and PRs that remain unresolved. These are the ones I've found from a very simple search for "absolute" and "relative". There are most likely others that are not accounted for here.
- Can not get right routes with multi sections #2370
- image url begin with "/" should be AbsolutePath,not RelativePath #2369
routeRewrite
/redirect
support likealias
#1982- Did not load homepage in router mode history and subdirectory #1763
- relativepath=true and _sidebar.md can't coexist #1761
- relative paths of links in sidebar should be relative to the sidebar file #1684
- using relativePath sidebar sublevels do not render if I use direct link to a page. #1631
- sidebar relative path issues #1569
- Unable to use absolute path for coverpage, loadNavbar, or loadSidebar options #1407
- “Embed files” doesn't effect #1033
- How to add images from root? #850
- When
relativePath: true
, even absolute paths are for icons in sidebar are resolved as relative. #877 - Same-level, sub-folder links, don't resolve properly #424
- Version 4.6.8 broke images (404) #415
Steps to reproduce
To help others understand the problem, I have created a demo link and non-link paths can be tested:
What is current behaviour
Test results from the Codesandbox Demo can be seen in the following table:
What is the expected behaviour
- Paths should be consistent for all resource types.
- All relative paths should be relative to the page in which they are used by default.
- All absolute paths should be relative to the top-most level in the domain or directory hierarchy by default.
- Configuration options should allow users to define base paths for relative and absolute links:
basePath
: Defaults tonull
. Functionally the same as settingrelativeBasePath
andabsoluteBasePath
to the same value.relativeBasePath
: Defaults to current page route. OverridesbasePath
.absoluteBasePath
: Defaults to/
. OverridesbasePath
.
- A
:basepath
markdown attribute (or similar) should allows specifying the base path for individual elements. This is required to allow accessing resources outside of the currentindex.html
route when one of the base path options above would otherwise prevent it. For example:[Root] ├─ assets │ └─ image.png └─ docs ├─ index.html └─ README.md
window.docsify = { basePath: '/docs/` }
Result:<!-- /docs/README.md -->  
<img src="/assets/image.png"> <img src="/assets/image.png">
If these changes are implemented, the default behavior will be as follows: