Before you start, ensure you have
- Created a GitHub account
- Signed the Particular Contributor License Agreement.
There are two approaches to contributing.
For simple changes, the GitHub web UI should suffice.
- Find the page you want to edit on http://docs.particular.net/.
- Click the
Improve this doc
button. This will automatically fork the project so you can edit the file. - Make the changes you require. Ensure you verify the changes in the
Preview
tab. - Add a description of the changes.
- Click
Propose File Changes
.
For more complex changes you should fork and then submit a pull request. This is useful if you are proposing multiple file changes
- Fork on GitHub.
- Clone the fork locally.
- Work on the feature.
- Push the code to GitHub.
- Send a Pull Request on GitHub.
For more information, see Collaborating on GitHub especially using GitHub pull requests.
If, as part of editing a page, a full review of the content is done, the reviewed header should be updated. This date is used to render http://docs.particular.net/review.
As part of a full review the following should be done:
- Spelling (US)
- Grammar
- Version specific language and content is correct
- Language is concise
- All links are relevant. No 3rd part links have redirects or 404s.
- Are there any more links that can be added to improve the content
- Content is correct up to and including the current released version
- Tags are correct
- Summary and title is adequate
Note that for minor changes (e.g. individual spelling or grammar fixes) the reviewed header should NOT be updated.
All content files (.md
, .png
, .jpg
etc) and directories must be lower case.
All links pointing to them must be lower case.
Use a dash (-
) to delimit filenames (e.g. specify-endpoint-name.md
).
Each document has a header. It is enclosed by ---
and is defined in a YAML document format.
The GitHub UI will correctly render YAML.
For example:
---
title: Auditing With NServiceBus
summary: Provides built-in message auditing for every endpoint.
versions: '[4,)'
tags:
- Auditing
- Forwarding Messages
related:
- samples/custom-checks/monitoring3rdparty
redirects:
- nservicebus/overview
---
title: Auditing With NServiceBus
Required. Used for the web page title tag <head><title>
, displayed in the page content, and displayed in search results.
versions: '[4,)'
Optional. Used for specifying what versions the given page covers, especially relevant for features that are not available in all supported versions. Format is 'nuget_version_range'
reviewed: 2016-03-01
Optional. Used to capture the last date that a page was fully reviewed. Format is yyyy-MM-dd
.
summary: Provides built-in message auditing for every endpoint.
Required. Used for the meta description tag (<meta name="description" />
) and displaying the search results.
tags:
- Auditing
- Forwarding Messages
Optional. Used to flag the article as being part of a group of articles.
Tags are rendered in the articles content with the full list of tags being rendered at http://docs.particular.net/tags. Untagged articles will be rendered here http://docs.particular.net/tags/untagged
Tags are interpreted in two ways.
- For inclusion in URLs:
- Tag are lower case
- Spaces are replaced with dashes (
-
)
- For display purposes:
- Tags are lower case
- Dashes (
-
) are replaced with spaces
Hidden
hidden: true
Causes two things:
- Stops search engines from finding the page using a
<meta name="robots" content="noindex" />
. - Prevents the page from being found in the docs search.
related:
- samples/custom-checks/monitoring3rdparty
A list of related pages for this page. These links will be rendered at the bottom of the page. Can include both samples and articles and they will be grouped as such when rendered in html.
redirects:
- nservicebus/overview
When renaming an existing article to a new name, add the redirects:
section in the article header and specify the previous name for the article. If the old URL is linked anywhere, the new renamed article will automatically be served when the user clicks on it.
- Values specified in the
redirects
section must be lower case. - Multiple values can be specified for the redirects, same as
tags
. - Values are fully qualified
Should be the URL relative to the root with no beginning or trailing slash padding and no .md.
- In the following example, whenever the URLs
/servicecontrol/sc-si
or/servicecontrol/debugging-servicecontrol
are being requested, the given article will be rendered.
---
title: ServiceInsight Interaction
summary: 'Using ServiceInsight Together'
tags:
- ServiceInsight
- Invocation
- Debugging
redirects:
- servicecontrol/sc-si
- servicecontrol/debugging-servicecontrol
related:
- samples/azure/shared-host
---
"Components" is a general term used to describe a deployable set of functionality. Components exist in components/components.yaml. Note that over time a component may have moved between nugets or split into new nugets. For example, the ABS DataBus or the Callbacks.
Sample Component:
- Key: Callbacks
Url: nservicebus/messaging/callbacks
NugetOrder:
- NServiceBus.Callbacks
- NServiceBus
The component key allows for shorthand when referring to components in page headers.
The component URL is the definitive source of documentation for a given component. This will eventually be used to link back to said documentation from both NuGet usages, samples and articles.
Since components can be split over multiple different nugets, it is not possible to infer the order from the NuGet version alone. So we need to have a lookup index and the NugetOrder allows us to sensibly sort component versions. For example, NServiceBus.Callbacks.1.0.0 should sort higher than the version of Callbacks that exists in NServiceBus.5.0.0.
All NServiceBus-related NuGet packages (used in documentation) are listed in components/nugetAlias.txt. The alias part of the NuGet is the key that is used to infer the version and component for all snippets. For example, Snippets/Callbacks has, over its lifetime, existed in both the Core NuGet and the Callbacks NuGet. So the directories under Callbacks are indicative of the NuGet (alias) they exist in and then split over the multiple versions of a given NuGet.
Example aliases:
ASP: NServiceBus.Persistence.AzureStorage
Autofac: NServiceBus.Autofac
Azure: NServiceBus.Azure
Callbacks: NServiceBus.Callbacks
The menu is a YAML text document stored at menu/menu.yaml.
Any sub-items that are prefixed with the title of the parent item will have that prefix removed
The directory structure where a .md
exists is used to derive the URL for that document.
So a file existing at nservicebus\logging\nlog.md
will have a URL of http://docs.particular.net/nservicebus/logging/nlog
.
One exception to the URL rule is when a page is named index.md
. In this case the index.md
is omitted in the URL and only the directory structure is used.
So a file existing at nservicebus\logging\index.md
will have a URL of http://docs.particular.net/nservicebus/logging/
.
Like any page an index page can include related pages. However index pages will, by default, have all sibling and child pages included in the list of related pages. This is effectively a recursive walk of the file system for the directory the given index.md exists in.
Links to other documentation pages should be relative and contain the .md
extension.
The .md
allows links to work inside the GitHub web UI. The .md
will be trimmed when they are finally rendered.
Given the case of editing a page located at \nservicebus\page1.md
:
- To link to the file
nservicebus\page2.md
, use[Page 2 Text](Page2.md)
. - To link to the file
\servicecontrol\page3.md
, use[Page 3 Text](/servicecontrol/page3.md)
.
Don't link to index.md
pages, instead link to the directory. So link to /nservicebus/logging
and NOT /nservicebus/logging/index.md
The site is rendered using GitHub Flavored Markdown
For editing markdown on the desktop (after cloning locally with Git) try MarkdownPad.
Ensure you enable GitHub Flavored Markdown (Offline)
by going to
Tools > Options > Markdown > Markdown Processor > GitHub Flavored Markdown (Offline)
Or click in the bottom left on the M
icon to "hot-switch"
Don't render YAML front-matter by going to
Tools > Options > Markdown > Markdown Settings
And checking Ignore YAML Front-matter
Any of the following, or combination thereof, could indicate that something should be a sample
- When there are multiple non-trivial moving pieces that would be mitigated by being able to download a runnable VS solution.
- When illustrating how Particular products/tools interact with 3rd-party products/tools.
- It is a sample of a significant feature of the Particular platform. e.g. Databus, encryption, pipeline etc.
Do not write a sample when:
- The only difference to an existing sample is a minor API usage.
- Samples should illustrate a feature or scenario with as few moving pieces as possible. For example, if the sample is "illustrating IOC with MVC" then "adding SignalR" to that sample will only cause confusion. In general, the fewer nugets you need to get the point across the better.
- Do not "document things inside a sample". A sample is to show how something is used, not to document it. Instead update the appropriate documentation page and link to it. As a general rule, if you add any content to a sample, where that guidance could possibly be applicable to other samples, then that guidance should probably exist in a documentation page.
- Samples are located here: https://github.com/Particular/docs.particular.net/tree/master/samples.
- They are linked to from the home page and are rendered here: http://docs.particular.net/samples/.
- Any directory in that structure with a sample.md will be considered a "root for a sample" or Sample Root.
- A Sample Root may not contain a sample.md in subdirectories.
- Each directory under the Sample Root will be rendered on the site as a downloadable zip with the directory name being the filename.
- A sample.md can use snippets from within its Sample Root but not snippets defined outside that root.
Startup projects are set to "all startable projects" in the solution. This is done via https://github.com/ParticularLabs/SetStartupProjects. If you want to control the default projects start then add a file named DefaultStartupProjects.txt
in the same directory as the solution file with relative paths to the project files you would like to use for startup projects.
For example if the solution contains two endpoints and you only want to start Endpoint1
the content of DefaultStartupProjects.txt
would be:
Endpoint1\Endpoint1.csproj
At the moment the best way to get started on a sample is to copy an existing one. Ideally one that is similar to what you are trying to achieve.
A good sample to start with is the Default Logging Sample, since all it does is enable logging. You can then add the various moving pieces to the copy.
Avoid using screenshots in samples unless it adds significant value over what can be expressed in text. They have the following problems:
- More time consuming to update than text
- Not search-able
- Prone to an inconsistent feel as different people take screenshots at different sizes, different zoom levels and with different color schemes for the app in question
- Add significantly to the page load time.
The most common misuse of screenshots is when capturing console output. DO NOT DO THIS. Put the text inside a formatted code section instead.
Partials are version specific files that contain markdown.
There are only rendered in the target page when the version filter matches the convention for a give file.
Partial Convention: filePrefix_key_nugetAlias_version.partial.md
The NuGet alias in samples should match the prefix as defined by the samples solution directories.
Partials are rendered in the target page by using the following syntax
partial: PARTIAL_KEY
So an example directory structure might be as follows
And to include the endpointname
partial can be pulled into sample.md
by including.
partial: endpointname
Markdown includes are pulled into the document prior to passing the content through the markdown conversion.
Add a file anywhere in the docs repository that is suffixed with .include.md
. For example, the file might be named theKey.include.md
.
Add the following to the markdown: include: theKey
There is a some code located here: https://github.com/Particular/docs.particular.net/tree/master/Snippets. Any directory containing an _excludesnippets
file will have its snippets ignored.
File extensions scanned for snippets include:
.config
.cs
.ps
.cscfg
.csdef
.html
.sql
.txt
.xml
- https://highlightjs.org/static/demo/
- https://github.com/isagalaev/highlight.js/tree/master/src/languages
language | key |
---|---|
c# | cs |
xml | xml |
no format | no-highlight |
command line | dos |
powershell | ps |
json | json |
sql | sql |
Any code wrapped in a convention-based comment will be picked up. The comment needs to start with startcode
which is followed by the key.
// startcode ConfigureWith
var configure = Configure.With();
// endcode
For non-code snippets apply a similar approach as in code, using comments appropriate for a given file type. For plain-text files an extra empty line is required before endcode
tag.
Tag | XML-based | PowerShell | SQL script | Plain text |
---|---|---|---|---|
Open | <!-- startcode name --> |
# startcode name |
-- startcode name |
startcode name |
Content | ||||
Close | <!-- endcode --> |
# endcode |
-- endcode |
endcode |
Any code wrapped in a named C# region will be picked up. The name of the region is used as the key.
#region ConfigureWith
var configure = Configure.With();
#endregion
Snippets are versioned. These versions are used to render snippets in a tabbed manner.
Versions follow the NuGet versioning convention. If either Minor
or Patch
is not defined they will be rendered as an x
. For example, Version 3.3
would be rendered as 3.3.x
and Version 3
would be rendered as 3.x
.
Snippet versions are derived in two ways
Appending a version to the end of a snippet definition as follows:
#region ConfigureWith 4.5
var configure = Configure.With();
#endregion
Or version range:
#region MySnippetName [1.0,2.0]
My Snippet Code
#endregion
If a snippet has no version defined then the version will be derived by walking up the directory tree until if finds a directory that is suffixed with _Version
or _VersionRange
. For example:
- Snippets extracted from
docs.particular.net\Snippets\Snippets_4\TheClass.cs
would have a default version of(≥ 4.0.0 && < 5.0.0)
. - Snippets extracted from
docs.particular.net\Snippets\Snippets_4\Special_4.3\TheClass.cs
would have a default version of(≥ 4.3.0 && < 5.0.0)
. - Snippets extracted from
docs.particular.net\Snippets\Special_(1.0,2.0)\TheClass.cs
would have a default version of(> 1.0.0 && < 2.0.0)
.
If a file named prerelease.txt
exists in a versioned directory then a -pre
will be added to the version.
For example, if there is a directory docs.particular.net\Snippets\Snippets_6\
and it contains a prerelease.txt
file then the version will be (≥ 6.0.0-pre)
The keyed snippets can then be used in any documentation .md
file by adding the text
snippet: KEY
Then snippets with the key (all versions) will be rendered in a tabbed manner. If there is only a single version then it will be rendered as a simple code block with no tabs.
For example:
To configure the bus call
snippet: ConfigureWith
The resulting markdown will be:
To configure the bus call
```
var configure = Configure.With();
```
The code snippets will do smart trimming of snippet indentation.
For example, given this snippet:
••#region DataBus ••var configure = Configure.With() ••••.FileShareDataBus(databusPath); ••#endregion
The two leading spaces (••) will be trimmed and the result will be
var configure = Configure.With()
••.FileShareDataBus(databusPath)
The same behavior will apply to leading tabs.
If tabs and spaces are mixed there is no way for the snippets to work out what to trim.
So given this snippet:
••#region DataBus ••var configure = Configure.With() ➙➙.FileShareDataBus(databusPath); ••#endregion
where ➙ is a tab, the resulting markdown will be
var configure = Configure.With() ➙➙.FileShareDataBus(databusPath)
Note that none of the tabs have been trimmed.
Use var
everywhere.
The code used by snippets and samples is compiled on the build server. The compilation is done against the versions of the packages referenced in the samples and snippets projects. When a snippet doesn't compile, the build will break so make sure snippets are compiling properly. Samples and snippets should not reference unreleased nugets.
There are some scenarios where documentation may require unreleased or beta NuGet packages. For example, when creating a PR against documentation for a feature that is not yet released. In this case, it is ok for a PR to reference an unreleased NuGet and have that PR fail to build on the build server. Once the NuGet packages have been released that PR can be merged.
In some cases it may be necessary to have merged documentation for unreleased features. In this case the NuGet packages should be pushed to the Particular feed on MyGet. The feed is included by default in the Snippets nuget.config.
Sometimes it is necessary to draw attention to items you want to call out in a document.
This is achieved through bootstrap alerts http://getbootstrap.com/components/#alerts
There are several keys each of which map to a different colored alert
Key | Color |
---|---|
SUCCESS |
green |
NOTE or INFO |
blue |
WARNING |
yellow |
DANGER |
red |
Keys can be used in two manners
This can be done with the following syntax
KEY: the note text.
For example, this
NOTE: Some sample note text.
will be rendered as
<p class="alert alert-info">
Some sample note text.
</p>
Sometimes it is necessary to group markdown elements inside a note. This can be done with the following syntax
{{KEY:
Inner markdown elements
}}
For example, this
{{NOTE:
* Point one
* Point Two
}}
will be rendered as
<p class="alert alert-info">
* Point One
* Point Two
</p>
The first (and all top level) headers in a .md
page should be a h2
(i.e. ##
) with sub-headings under it being h3
, h4
, etc.
- Two empty lines before a heading and any other text
- Add an empty line after a heading
- Add an empty line between paragraphs
One addition to standard markdown is the auto creation of anchors for headings.
So if you have a heading like this:
## My Heading
it will be converted to this:
<h2>
<a name="my-heading"/>
My Heading
</h2>
Which means elsewhere in the page you can link to it with this:
[Goto My Heading](#My-Heading)
Images can be added using the following markdown syntax
![Alt text](/path/to/img.jpg "Optional title")
With the minimal syntax being
![](/path/to/img.jpg)
Image size can be controlled by adding the text width=x
to the end of the title
For example
![Alt text](/path/to/img.jpg "Optional title width=x")
With the minimal syntax being
![](/path/to/img.jpg "width=x")
This will result in the image being re-sized with the following parameters
width="x" height="auto"
It will also wrap the image in a clickable lightbox so the full image can be accessed.
When creating images, strive to keep sources in order to update and re-create images later. For diagrams, LucidChart can be used. LucidChart allows export and import of Visio (VDX) formatted documents. Visio formatted documents can be used to generate images and should be committed along with the images. To generate images from LucidChart (or a Visio document), export the image as PNG, using the "Crop to content" option.
Sequence diagram images are generated using the https://bramp.github.io/js-sequence-diagrams/ online service. Keep the source text used to generate sequence image in the document as an HTML comment to allow future modifications in case images need to be re-generated.
Flowchar diagrams can be generated with Code2Flow using a nice pseudocode syntax like this:
switch (Operation?) {
case Send:
Determine routing for `Send`;
break;
case Publish:
Determine routing for `Publish`;
break;
case Reply:
Determine routing for `Reply`;
break;
}
For a more complex example, go to the routing page.
- Ticks are done with
✔
✔ - Crosses are done with
✖
✖
For consistency, prefer American English.
Avoid personal voice. I.e. no "we", "you", "your", "our" etc.
Avoid ambiguity.
Versions X and above and Versions Y and below and Version X to Version Y.
Versions X and NOT VX or version X.
The word Bus
should be avoided in documentation. Some replacements include:
- When referring to the topology, use
federated
(for which the opposite term iscentralized
) - When referring to the NServiceBus instance, the general thing that sends or publishes messages, use
endpoint instance
orendpoint
(when it is clear from the context that you are talking about an instance rather than a logical concept) - When referring specifically to the
IBus
interface usemessage session
ormessage context
(depending if you are talking about just sending a messages from external component or from inside a handler)
The word Bus
is allowed when a particular piece of documentation refers specifically to version 5 or below and discusses low level implementation details.
Avoid deep link into the RavenDB documentation since it is a maintenance pain. For example don't link to http://ravendb.net/docs/article-page/3.0/Csharp/client-api//session/transaction-support/dtc-transactions#transaction-storage-recovery
since when RavenDB 4 is release article-page/3.0/Csharp
is invalid and requires an update. Also the RavenDB documentation does not maintain structure between versions. e.g. http://ravendb.net/docs/article-page/2.0/Csharp/client-api//session/transaction-support/dtc-transactions#transaction-storage-recovery
is a 404. So we can't trust that "just change the version" will work. Instead link to the RavenDB docs search: http://ravendb.net/docs/search/latest/csharp?searchTerm=THE-SEARCH-TERM
. So for the above example it would be http://ravendb.net/docs/search/latest/csharp?searchTerm=Transaction-storage-recovery
.
Under https://github.com/Particular/docs.particular.net/tree/master/tools there are several utilities to help with the management of this repository. All are in the form of LINQPad scripts.
Uses nuget.exe to update all NuGet packages in all solutions to the newest patch version. This script takes 10-20 minutes depending on bandwidth.
Enforces the Resharper settings to be correct for every solution. The standard is a placeholder .settings file that pull in the Shared.DotSettings file as a layer.
Sets the correct startup projects for every solution. This is persisted in an .suo file for each solution. Since .suo files are not committed to source control, if a re-clone is done this script will need to be re-run.