This is a blog software completely written in C# / Blazor. The aim is to have it configurable as possible.
The basic idea is that the content creator writes his posts in markdown language (like this readme file). The markdown will then be translated into HTML and displayed to the client. This gives an easy entry to writing posts with all the flexibility markdown has. This also includes source code snippets. Right now only C# is highlighted properly but other languages can be extended easily as the highlighting is done via highlight.js.
Just clone this repository and you are good to go. There are some settings you can tweak. The following chapter will guide you through the possibilities.
The appsettings.json file has a lot of options to customize the content of the blog. The following table shows which values are used when.
{
"BlogName": "linkdotnet",
"GithubAccountUrl": "",
"LinkedInAccountUrl": "",
"Introduction": {
"Description": "Some nice text about yourself. Markup can be used [Github](https://github.com/someuser/somerepo)",
"BackgroundUrl": "assets/profile-background.webp",
"ProfilePictureUrl": "assets/profile-picture.webp"
},
"PersistenceProvider": "SqlServer",
"ConnectionString": "",
"DatabaseName": "",
"Auth0": {
"Domain": "",
"ClientId": "",
"ClientSecret": ""
},
"BlogPostsPerPage": 10,
"AboutMeProfileInformation": {
"Name": "Steven Giesel",
"Heading": "Software Engineer",
"ProfilePictureUrl": "assets/profile-picture.webp"
},
"Giscus": {
"Repository": "github/repo",
"RepositoryId": "id",
"Category": "general",
"CategoryId": "id"
},
"Disqus": {
"Shortname": "blog"
}
}
Property | Type | Description |
---|---|---|
BlogName | string | Name of your blog. Is used in the navbar and is used as the title of the page. |
GithubAccountUrl | string | Url to your github account. If not set the navigation link is not shown |
LinkedInAccountUrl | string | Url to your LinkedIn account. If not set the navigation link is not shown |
Introduction | Is used for the introduction part of the blog | |
Description | MarkdownString | Small introduction text for yourself. This is also used for <meta name="description"> tag. For this the markup will be converted to plain text. |
BackgroundUrl | string | Url or path to the background image |
ProfilePictureUrl | string | Url or path to your profile picture |
PersistenceProvider | string | Declares the type of the storage provider (one of the following: SqlServer , SqliteServer , RavenDb , InMemory ). More in-depth explanation down below |
ConnectionString | string | Is used for connection to a database. Not used when InMemoryStorageProvider is used |
DatabaseName | string | Name of the database. Only used with RavenDbStorageProvider |
Auth0 | Configuration for setting up Auth0 | |
Domain | string | See more details here: https://manage.auth0.com/dashboard/ |
ClientId | string | See more details here: https://manage.auth0.com/dashboard/ |
ClientSecret | string | See more details here: https://manage.auth0.com/dashboard/ |
BlogPostsPerPage | int | Gives the amount of blog posts loaded and display per page. For more the user has to use the navigation |
AboutMeProfileInformation | node | Sets information for the About Me Page. If omitted the page is disabled completely |
Name | string | Name, which is displayed on top of the profile card |
Heading | string | Displayed under the name. For example job title |
ProfilePictureUrl | string | Displayed profile picture |
Giscus | node | Enables the comment section via giscus. If left empty the comment secion will not be shown. For more information checkout the section about comments down below |
Disqus | node | Enables the comment section via disqus. If left empty the comment secion will not be shown. For more information checkout the section about comments down below |
Currently there are 4 Storage-Provider:
- InMemory - Basically a list holding your data (per request). If the User hits a hard reload, the data is gone.
- RavenDb - As the name suggests for RavenDb
- Sqlite - Based on EF Core, so it can be easily adapted for other Sql Dialects
- SqlServer - Based on EF Core, so it can be easily adapted for other Sql Dialects
For comments the blog is using giscus or disqus.
Note: Please only register one service / provide the configuration for one node. The given configuration shows both only for demonstrational purposes. If multiple comment plugins are configured at the same time a warning will be displayed under the blog post itself.
To provide the necessary values head over to https://giscus.app/ and go to the configuration section. There you can enter all the information. You will find a detailed guide on the site.
In short:
- You need a public repository where the comments are hosted. Recommendation: Create a new repository just for the comments
- You have to link the giscus app to at least the repository where the comments are hosted
- You have to enable the discussion feature in the repository (see here )
After you configured everything on the site, you get the <script>
tag which you could embed. The blog needs the following information.
Here you can find an example. This is how the script tag looks on giscus.
<script src="https://giscus.app/client.js"
data-repo="your_username/reponame"
data-repo-id="M9/ab=="
data-category="General"
data-category-id="AbC==/8_D"
async>
</script>
Now you can copy/paste those information into the appsettings.json. With the given information abobe your appsettings.json looks like this:
"Giscus": {
"Repository": "your_username/reponame",
"RepositoryId": "M9/ab==",
"Category": "General",
"CategoryId": "AbC==/8_D"
}
For disqus you only need the shortname (site-name) which you can find for example under your home-tab.
There is only one real mechanism enabled via Auth0. For more information go to: https://auth0.com/docs/applications
The main advantage of Auth0 is the easy configurable dashboard on their website.
For testing purposes you can use services.UseDummyAuthentication();
. This allows every user, who presses Login, to be logged in.
The blog includes some of the most important tags to get indexed by a crawler. Furthermore some aspects of the Open Graph specification are implemented.
In the wwwroot/ you can find a default robots.txt. It allows that the site gets completely indexed. If you want to tweak that behavior feel free.
Also you can provide a sitemap.xml to get a better ranking. The blog can create a sitemap.xml on its own. For that login and click on the Admin
button in the navigation bar and afterwards on Sitemap
. There you can let the blog create a new one for you. This is especially helpful after you created a new blog post to make easier for indexer like Google.
To get better results when for example shared via LinkedIn some of the <meta property="og:tag">
tags are implemented.
The following tags are set depending on the page:
Open Graph Tag | Index | Display Blog Post |
---|---|---|
og:title | Title of the blog (defined in Introduction) | Title of the Blog Post |
og:url | Url to the index page | Url of the page |
og:image | Background image (defined in Introduction) | Yes |
og:type | article | article |
og:description | Short description in plain text (defined in Introduction) | Short Description of Blog Post in plain text |
Furthermore the following tags are set:
Tag | Index | Display Blog Post |
---|---|---|
Title of the web page | Defined in AppConfiguration.BlogName | Title of the blogpost |
<meta name="keyword" content="" /> | not set | Tags defined in the Blog Post |