feat: initial development
This commit is contained in:
208
.editorconfig
Normal file
208
.editorconfig
Normal file
@@ -0,0 +1,208 @@
|
|||||||
|
# To learn more about .editorconfig see https://aka.ms/editorconfigdocs
|
||||||
|
|
||||||
|
# top-most EditorConfig file
|
||||||
|
root = true
|
||||||
|
|
||||||
|
# Default settings:
|
||||||
|
# A newline ending every file
|
||||||
|
# Use 4 spaces as indentation
|
||||||
|
[*]
|
||||||
|
insert_final_newline = true
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 4
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
|
||||||
|
[*.json]
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
[*.{cs,xaml}]
|
||||||
|
charset = utf-8
|
||||||
|
end_of_line = crlf
|
||||||
|
|
||||||
|
# C# files
|
||||||
|
[*.cs]
|
||||||
|
# New line preferences
|
||||||
|
csharp_new_line_before_open_brace = all
|
||||||
|
csharp_new_line_before_else = true
|
||||||
|
csharp_new_line_before_catch = true
|
||||||
|
csharp_new_line_before_finally = true
|
||||||
|
csharp_new_line_before_members_in_object_initializers = true
|
||||||
|
csharp_new_line_before_members_in_anonymous_types = true
|
||||||
|
csharp_new_line_between_query_expression_clauses = true
|
||||||
|
|
||||||
|
# Indentation preferences
|
||||||
|
csharp_indent_block_contents = true
|
||||||
|
csharp_indent_braces = false
|
||||||
|
csharp_indent_case_contents = true
|
||||||
|
csharp_indent_case_contents_when_block = true
|
||||||
|
csharp_indent_switch_labels = true
|
||||||
|
csharp_indent_labels = one_less_than_current
|
||||||
|
|
||||||
|
# Parentheses preferences
|
||||||
|
dotnet_style_parentheses_in_arithmetic_binary_operators = never_if_unnecessary : silent
|
||||||
|
dotnet_style_parentheses_in_relational_binary_operators = never_if_unnecessary : silent
|
||||||
|
dotnet_style_parentheses_in_other_binary_operators = never_if_unnecessary : silent
|
||||||
|
dotnet_style_parentheses_in_other_operators = never_if_unnecessary : silent
|
||||||
|
|
||||||
|
# Modifier preferences
|
||||||
|
csharp_preferred_modifier_order = public,private,protected,internal,static,extern,new,virtual,abstract,sealed,override,readonly,unsafe,volatile,async : suggestion
|
||||||
|
dotnet_style_require_accessibility_modifiers = for_non_interface_members : warning
|
||||||
|
|
||||||
|
# avoid this. unless absolutely necessary
|
||||||
|
dotnet_style_qualification_for_field = false : suggestion
|
||||||
|
dotnet_style_qualification_for_property = false : suggestion
|
||||||
|
dotnet_style_qualification_for_method = false : suggestion
|
||||||
|
dotnet_style_qualification_for_event = false : suggestion
|
||||||
|
|
||||||
|
# Types: use keywords instead of BCL types, and permit var only when the type is clear
|
||||||
|
csharp_style_var_for_built_in_types = false : suggestion
|
||||||
|
csharp_style_var_when_type_is_apparent = false : none
|
||||||
|
csharp_style_var_elsewhere = false : suggestion
|
||||||
|
dotnet_style_predefined_type_for_locals_parameters_members = true : suggestion
|
||||||
|
dotnet_style_predefined_type_for_member_access = true : suggestion
|
||||||
|
|
||||||
|
# Naming Conventions
|
||||||
|
|
||||||
|
dotnet_naming_style.pascal_case_style.capitalization = pascal_case
|
||||||
|
|
||||||
|
dotnet_naming_style.camel_case_underscore_style.capitalization = camel_case
|
||||||
|
dotnet_naming_style.camel_case_underscore_style.required_prefix = _
|
||||||
|
|
||||||
|
dotnet_naming_style.interfaces_style.capitalization = pascal_case
|
||||||
|
dotnet_naming_style.interfaces_style.required_prefix = I
|
||||||
|
|
||||||
|
# name all constant fields using PascalCase
|
||||||
|
dotnet_naming_rule.constant_fields_should_be_pascal_case.severity = suggestion
|
||||||
|
dotnet_naming_rule.constant_fields_should_be_pascal_case.symbols = constant_fields
|
||||||
|
dotnet_naming_rule.constant_fields_should_be_pascal_case.style = pascal_case_style
|
||||||
|
dotnet_naming_symbols.constant_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.constant_fields.required_modifiers = const
|
||||||
|
|
||||||
|
# static fields using PascalCase
|
||||||
|
dotnet_naming_rule.static_fields_should_have_prefix.severity = suggestion
|
||||||
|
dotnet_naming_rule.static_fields_should_have_prefix.symbols = static_fields
|
||||||
|
dotnet_naming_rule.static_fields_should_have_prefix.style = pascal_case_style
|
||||||
|
dotnet_naming_symbols.static_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.static_fields.required_modifiers = static
|
||||||
|
dotnet_naming_symbols.static_fields.applicable_accessibilities = private, internal, private_protected
|
||||||
|
|
||||||
|
# internal and private fields should be _camelCase
|
||||||
|
dotnet_naming_rule.camel_case_for_private_internal_fields.severity = suggestion
|
||||||
|
dotnet_naming_rule.camel_case_for_private_internal_fields.symbols = private_internal_fields
|
||||||
|
dotnet_naming_rule.camel_case_for_private_internal_fields.style = camel_case_underscore_style
|
||||||
|
dotnet_naming_symbols.private_internal_fields.applicable_kinds = field
|
||||||
|
dotnet_naming_symbols.private_internal_fields.applicable_accessibilities = private, internal
|
||||||
|
|
||||||
|
# interfaces must have an I prefix
|
||||||
|
dotnet_naming_rule.interfaces_must_be_interfaces_style.severity = suggestion
|
||||||
|
dotnet_naming_rule.interfaces_must_be_interfaces_style.symbols = interfaces
|
||||||
|
dotnet_naming_rule.interfaces_must_be_interfaces_style.style = interfaces_style
|
||||||
|
dotnet_naming_symbols.interfaces.applicable_kinds = interface
|
||||||
|
dotnet_naming_symbols.interfaces.applicable_accessibilities = *
|
||||||
|
|
||||||
|
# Code style defaults
|
||||||
|
csharp_using_directive_placement = outside_namespace : suggestion
|
||||||
|
dotnet_sort_system_directives_first = true
|
||||||
|
csharp_prefer_braces = true : suggestion
|
||||||
|
csharp_preserve_single_line_blocks = true : none
|
||||||
|
csharp_preserve_single_line_statements = false : none
|
||||||
|
csharp_prefer_static_local_function = true : suggestion
|
||||||
|
csharp_prefer_simple_using_statement = false : none
|
||||||
|
csharp_style_prefer_switch_expression = true : suggestion
|
||||||
|
|
||||||
|
# Code quality
|
||||||
|
dotnet_style_readonly_field = true : suggestion
|
||||||
|
dotnet_code_quality_unused_parameters = non_public : suggestion
|
||||||
|
|
||||||
|
# Expression-level preferences
|
||||||
|
dotnet_style_object_initializer = true : suggestion
|
||||||
|
dotnet_style_collection_initializer = true : suggestion
|
||||||
|
dotnet_style_explicit_tuple_names = true : suggestion
|
||||||
|
dotnet_style_coalesce_expression = true : suggestion
|
||||||
|
dotnet_style_null_propagation = true : suggestion
|
||||||
|
dotnet_style_prefer_is_null_check_over_reference_equality_method = true : suggestion
|
||||||
|
dotnet_style_prefer_inferred_tuple_names = true : suggestion
|
||||||
|
dotnet_style_prefer_inferred_anonymous_type_member_names = true : suggestion
|
||||||
|
dotnet_style_prefer_auto_properties = true : suggestion
|
||||||
|
dotnet_style_prefer_conditional_expression_over_assignment = true : refactoring
|
||||||
|
dotnet_style_prefer_conditional_expression_over_return = true : refactoring
|
||||||
|
csharp_prefer_simple_default_expression = true : suggestion
|
||||||
|
csharp_style_deconstructed_variable_declaration = true : suggestion
|
||||||
|
|
||||||
|
# Expression-bodied members
|
||||||
|
csharp_style_expression_bodied_methods = true : refactoring
|
||||||
|
csharp_style_expression_bodied_constructors = true : refactoring
|
||||||
|
csharp_style_expression_bodied_operators = true : refactoring
|
||||||
|
csharp_style_expression_bodied_properties = true : refactoring
|
||||||
|
csharp_style_expression_bodied_indexers = true : refactoring
|
||||||
|
csharp_style_expression_bodied_accessors = true : refactoring
|
||||||
|
csharp_style_expression_bodied_lambdas = true : refactoring
|
||||||
|
csharp_style_expression_bodied_local_functions = true : refactoring
|
||||||
|
|
||||||
|
# Pattern matching
|
||||||
|
csharp_style_pattern_matching_over_is_with_cast_check = true : suggestion
|
||||||
|
csharp_style_pattern_matching_over_as_with_null_check = true : suggestion
|
||||||
|
csharp_style_inlined_variable_declaration = true : suggestion
|
||||||
|
|
||||||
|
# Null checking preferences
|
||||||
|
csharp_style_throw_expression = true : suggestion
|
||||||
|
csharp_style_conditional_delegate_call = true : suggestion
|
||||||
|
|
||||||
|
# Other features
|
||||||
|
csharp_style_prefer_index_operator = false : none
|
||||||
|
csharp_style_prefer_range_operator = false : none
|
||||||
|
csharp_style_pattern_local_over_anonymous_function = false : none
|
||||||
|
|
||||||
|
# Space preferences
|
||||||
|
csharp_space_after_cast = false
|
||||||
|
csharp_space_after_colon_in_inheritance_clause = true
|
||||||
|
csharp_space_after_comma = true
|
||||||
|
csharp_space_after_dot = false
|
||||||
|
csharp_space_after_keywords_in_control_flow_statements = true
|
||||||
|
csharp_space_after_semicolon_in_for_statement = true
|
||||||
|
csharp_space_around_binary_operators = before_and_after
|
||||||
|
csharp_space_around_declaration_statements = do_not_ignore
|
||||||
|
csharp_space_before_colon_in_inheritance_clause = true
|
||||||
|
csharp_space_before_comma = false
|
||||||
|
csharp_space_before_dot = false
|
||||||
|
csharp_space_before_open_square_brackets = false
|
||||||
|
csharp_space_before_semicolon_in_for_statement = false
|
||||||
|
csharp_space_between_empty_square_brackets = false
|
||||||
|
csharp_space_between_method_call_empty_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_method_call_name_and_opening_parenthesis = false
|
||||||
|
csharp_space_between_method_call_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_method_declaration_empty_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_method_declaration_name_and_open_parenthesis = false
|
||||||
|
csharp_space_between_method_declaration_parameter_list_parentheses = false
|
||||||
|
csharp_space_between_parentheses = false
|
||||||
|
csharp_space_between_square_brackets = false
|
||||||
|
|
||||||
|
# Analyzers
|
||||||
|
dotnet_code_quality.ca1802.api_surface = private, internal
|
||||||
|
|
||||||
|
# C++ Files
|
||||||
|
[*.{cpp,h,in}]
|
||||||
|
curly_bracket_next_line = true
|
||||||
|
indent_brace_style = Allman
|
||||||
|
|
||||||
|
# Xml project files
|
||||||
|
[*.{csproj,vcxproj,vcxproj.filters,proj,nativeproj,locproj}]
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
# Xml build files
|
||||||
|
[*.builds]
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
# Xml files
|
||||||
|
[*.{xml,stylecop,resx,ruleset}]
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
# Xml config files
|
||||||
|
[*.{props,targets,config,nuspec}]
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
# Shell scripts
|
||||||
|
[*.sh]
|
||||||
|
end_of_line = lf
|
||||||
|
[*.{cmd, bat}]
|
||||||
|
end_of_line = crlf
|
||||||
2
.gitattributes
vendored
Normal file
2
.gitattributes
vendored
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
# Auto detect text files and perform LF normalization
|
||||||
|
* text=auto
|
||||||
334
.gitignore
vendored
Normal file
334
.gitignore
vendored
Normal file
@@ -0,0 +1,334 @@
|
|||||||
|
## Ignore Visual Studio temporary files, build results, and
|
||||||
|
## files generated by popular Visual Studio add-ons.
|
||||||
|
##
|
||||||
|
## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
|
||||||
|
|
||||||
|
# User-specific files
|
||||||
|
*.rsuser
|
||||||
|
*.suo
|
||||||
|
*.user
|
||||||
|
*.userosscache
|
||||||
|
*.sln.docstates
|
||||||
|
|
||||||
|
# User-specific files (MonoDevelop/Xamarin Studio)
|
||||||
|
*.userprefs
|
||||||
|
|
||||||
|
# Build results
|
||||||
|
[Dd]ebug/
|
||||||
|
[Dd]ebugPublic/
|
||||||
|
[Rr]elease/
|
||||||
|
[Rr]eleases/
|
||||||
|
x64/
|
||||||
|
x86/
|
||||||
|
bld/
|
||||||
|
[Bb]in/
|
||||||
|
[Oo]bj/
|
||||||
|
[Ll]og/
|
||||||
|
|
||||||
|
# Visual Studio 2015/2017 cache/options directory
|
||||||
|
.vs/
|
||||||
|
# Uncomment if you have tasks that create the project's static files in wwwroot
|
||||||
|
#wwwroot/
|
||||||
|
|
||||||
|
# Visual Studio 2017 auto generated files
|
||||||
|
Generated\ Files/
|
||||||
|
|
||||||
|
# MSTest test Results
|
||||||
|
[Tt]est[Rr]esult*/
|
||||||
|
[Bb]uild[Ll]og.*
|
||||||
|
|
||||||
|
# NUNIT
|
||||||
|
*.VisualState.xml
|
||||||
|
TestResult.xml
|
||||||
|
|
||||||
|
# Build Results of an ATL Project
|
||||||
|
[Dd]ebugPS/
|
||||||
|
[Rr]eleasePS/
|
||||||
|
dlldata.c
|
||||||
|
|
||||||
|
# Benchmark Results
|
||||||
|
BenchmarkDotNet.Artifacts/
|
||||||
|
|
||||||
|
# .NET Core
|
||||||
|
project.lock.json
|
||||||
|
project.fragment.lock.json
|
||||||
|
artifacts/
|
||||||
|
|
||||||
|
# StyleCop
|
||||||
|
StyleCopReport.xml
|
||||||
|
|
||||||
|
# Files built by Visual Studio
|
||||||
|
*_i.c
|
||||||
|
*_p.c
|
||||||
|
*_h.h
|
||||||
|
*.ilk
|
||||||
|
*.meta
|
||||||
|
*.obj
|
||||||
|
*.iobj
|
||||||
|
*.pch
|
||||||
|
*.pdb
|
||||||
|
*.ipdb
|
||||||
|
*.pgc
|
||||||
|
*.pgd
|
||||||
|
*.rsp
|
||||||
|
*.sbr
|
||||||
|
*.tlb
|
||||||
|
*.tli
|
||||||
|
*.tlh
|
||||||
|
*.tmp
|
||||||
|
*.tmp_proj
|
||||||
|
*_wpftmp.csproj
|
||||||
|
*.log
|
||||||
|
*.vspscc
|
||||||
|
*.vssscc
|
||||||
|
.builds
|
||||||
|
*.pidb
|
||||||
|
*.svclog
|
||||||
|
*.scc
|
||||||
|
|
||||||
|
# Chutzpah Test files
|
||||||
|
_Chutzpah*
|
||||||
|
|
||||||
|
# Visual C++ cache files
|
||||||
|
ipch/
|
||||||
|
*.aps
|
||||||
|
*.ncb
|
||||||
|
*.opendb
|
||||||
|
*.opensdf
|
||||||
|
*.sdf
|
||||||
|
*.cachefile
|
||||||
|
*.VC.db
|
||||||
|
*.VC.VC.opendb
|
||||||
|
|
||||||
|
# Visual Studio profiler
|
||||||
|
*.psess
|
||||||
|
*.vsp
|
||||||
|
*.vspx
|
||||||
|
*.sap
|
||||||
|
|
||||||
|
# Visual Studio Trace Files
|
||||||
|
*.e2e
|
||||||
|
|
||||||
|
# TFS 2012 Local Workspace
|
||||||
|
$tf/
|
||||||
|
|
||||||
|
# Guidance Automation Toolkit
|
||||||
|
*.gpState
|
||||||
|
|
||||||
|
# ReSharper is a .NET coding add-in
|
||||||
|
_ReSharper*/
|
||||||
|
*.[Rr]e[Ss]harper
|
||||||
|
*.DotSettings.user
|
||||||
|
|
||||||
|
# JustCode is a .NET coding add-in
|
||||||
|
.JustCode
|
||||||
|
|
||||||
|
# TeamCity is a build add-in
|
||||||
|
_TeamCity*
|
||||||
|
|
||||||
|
# DotCover is a Code Coverage Tool
|
||||||
|
*.dotCover
|
||||||
|
|
||||||
|
# AxoCover is a Code Coverage Tool
|
||||||
|
.axoCover/*
|
||||||
|
!.axoCover/settings.json
|
||||||
|
|
||||||
|
# Visual Studio code coverage results
|
||||||
|
*.coverage
|
||||||
|
*.coveragexml
|
||||||
|
|
||||||
|
# NCrunch
|
||||||
|
_NCrunch_*
|
||||||
|
.*crunch*.local.xml
|
||||||
|
nCrunchTemp_*
|
||||||
|
|
||||||
|
# MightyMoose
|
||||||
|
*.mm.*
|
||||||
|
AutoTest.Net/
|
||||||
|
|
||||||
|
# Web workbench (sass)
|
||||||
|
.sass-cache/
|
||||||
|
|
||||||
|
# Installshield output folder
|
||||||
|
[Ee]xpress/
|
||||||
|
|
||||||
|
# DocProject is a documentation generator add-in
|
||||||
|
DocProject/buildhelp/
|
||||||
|
DocProject/Help/*.HxT
|
||||||
|
DocProject/Help/*.HxC
|
||||||
|
DocProject/Help/*.hhc
|
||||||
|
DocProject/Help/*.hhk
|
||||||
|
DocProject/Help/*.hhp
|
||||||
|
DocProject/Help/Html2
|
||||||
|
DocProject/Help/html
|
||||||
|
|
||||||
|
# Click-Once directory
|
||||||
|
publish/
|
||||||
|
|
||||||
|
# Publish Web Output
|
||||||
|
*.[Pp]ublish.xml
|
||||||
|
*.azurePubxml
|
||||||
|
# Note: Comment the next line if you want to checkin your web deploy settings,
|
||||||
|
# but database connection strings (with potential passwords) will be unencrypted
|
||||||
|
*.pubxml
|
||||||
|
*.publishproj
|
||||||
|
|
||||||
|
# Microsoft Azure Web App publish settings. Comment the next line if you want to
|
||||||
|
# checkin your Azure Web App publish settings, but sensitive information contained
|
||||||
|
# in these scripts will be unencrypted
|
||||||
|
PublishScripts/
|
||||||
|
|
||||||
|
# NuGet Packages
|
||||||
|
*.nupkg
|
||||||
|
# The packages folder can be ignored because of Package Restore
|
||||||
|
**/[Pp]ackages/*
|
||||||
|
# except build/, which is used as an MSBuild target.
|
||||||
|
!**/[Pp]ackages/build/
|
||||||
|
# Uncomment if necessary however generally it will be regenerated when needed
|
||||||
|
#!**/[Pp]ackages/repositories.config
|
||||||
|
# NuGet v3's project.json files produces more ignorable files
|
||||||
|
*.nuget.props
|
||||||
|
*.nuget.targets
|
||||||
|
|
||||||
|
# Microsoft Azure Build Output
|
||||||
|
csx/
|
||||||
|
*.build.csdef
|
||||||
|
|
||||||
|
# Microsoft Azure Emulator
|
||||||
|
ecf/
|
||||||
|
rcf/
|
||||||
|
|
||||||
|
# Windows Store app package directories and files
|
||||||
|
AppPackages/
|
||||||
|
BundleArtifacts/
|
||||||
|
Package.StoreAssociation.xml
|
||||||
|
_pkginfo.txt
|
||||||
|
*.appx
|
||||||
|
|
||||||
|
# Visual Studio cache files
|
||||||
|
# files ending in .cache can be ignored
|
||||||
|
*.[Cc]ache
|
||||||
|
# but keep track of directories ending in .cache
|
||||||
|
!*.[Cc]ache/
|
||||||
|
|
||||||
|
# Others
|
||||||
|
ClientBin/
|
||||||
|
~$*
|
||||||
|
*~
|
||||||
|
*.dbmdl
|
||||||
|
*.dbproj.schemaview
|
||||||
|
*.jfm
|
||||||
|
*.pfx
|
||||||
|
*.publishsettings
|
||||||
|
orleans.codegen.cs
|
||||||
|
|
||||||
|
# Including strong name files can present a security risk
|
||||||
|
# (https://github.com/github/gitignore/pull/2483#issue-259490424)
|
||||||
|
#*.snk
|
||||||
|
|
||||||
|
# Since there are multiple workflows, uncomment next line to ignore bower_components
|
||||||
|
# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
|
||||||
|
#bower_components/
|
||||||
|
|
||||||
|
# RIA/Silverlight projects
|
||||||
|
Generated_Code/
|
||||||
|
|
||||||
|
# Backup & report files from converting an old project file
|
||||||
|
# to a newer Visual Studio version. Backup files are not needed,
|
||||||
|
# because we have git ;-)
|
||||||
|
_UpgradeReport_Files/
|
||||||
|
Backup*/
|
||||||
|
UpgradeLog*.XML
|
||||||
|
UpgradeLog*.htm
|
||||||
|
ServiceFabricBackup/
|
||||||
|
*.rptproj.bak
|
||||||
|
|
||||||
|
# SQL Server files
|
||||||
|
*.mdf
|
||||||
|
*.ldf
|
||||||
|
*.ndf
|
||||||
|
|
||||||
|
# Business Intelligence projects
|
||||||
|
*.rdl.data
|
||||||
|
*.bim.layout
|
||||||
|
*.bim_*.settings
|
||||||
|
*.rptproj.rsuser
|
||||||
|
|
||||||
|
# Microsoft Fakes
|
||||||
|
FakesAssemblies/
|
||||||
|
|
||||||
|
# GhostDoc plugin setting file
|
||||||
|
*.GhostDoc.xml
|
||||||
|
|
||||||
|
# Node.js Tools for Visual Studio
|
||||||
|
.ntvs_analysis.dat
|
||||||
|
node_modules/
|
||||||
|
|
||||||
|
# Visual Studio 6 build log
|
||||||
|
*.plg
|
||||||
|
|
||||||
|
# Visual Studio 6 workspace options file
|
||||||
|
*.opt
|
||||||
|
|
||||||
|
# Visual Studio 6 auto-generated workspace file (contains which files were open etc.)
|
||||||
|
*.vbw
|
||||||
|
|
||||||
|
# Visual Studio LightSwitch build output
|
||||||
|
**/*.HTMLClient/GeneratedArtifacts
|
||||||
|
**/*.DesktopClient/GeneratedArtifacts
|
||||||
|
**/*.DesktopClient/ModelManifest.xml
|
||||||
|
**/*.Server/GeneratedArtifacts
|
||||||
|
**/*.Server/ModelManifest.xml
|
||||||
|
_Pvt_Extensions
|
||||||
|
|
||||||
|
# Paket dependency manager
|
||||||
|
.paket/paket.exe
|
||||||
|
paket-files/
|
||||||
|
|
||||||
|
# FAKE - F# Make
|
||||||
|
.fake/
|
||||||
|
|
||||||
|
# JetBrains Rider
|
||||||
|
.idea/
|
||||||
|
*.sln.iml
|
||||||
|
|
||||||
|
# CodeRush personal settings
|
||||||
|
.cr/personal
|
||||||
|
|
||||||
|
# Python Tools for Visual Studio (PTVS)
|
||||||
|
__pycache__/
|
||||||
|
*.pyc
|
||||||
|
|
||||||
|
# Cake - Uncomment if you are using it
|
||||||
|
# tools/**
|
||||||
|
# !tools/packages.config
|
||||||
|
|
||||||
|
# Tabs Studio
|
||||||
|
*.tss
|
||||||
|
|
||||||
|
# Telerik's JustMock configuration file
|
||||||
|
*.jmconfig
|
||||||
|
|
||||||
|
# BizTalk build output
|
||||||
|
*.btp.cs
|
||||||
|
*.btm.cs
|
||||||
|
*.odx.cs
|
||||||
|
*.xsd.cs
|
||||||
|
|
||||||
|
# OpenCover UI analysis results
|
||||||
|
OpenCover/
|
||||||
|
|
||||||
|
# Azure Stream Analytics local run output
|
||||||
|
ASALocalRun/
|
||||||
|
|
||||||
|
# MSBuild Binary and Structured Log
|
||||||
|
*.binlog
|
||||||
|
|
||||||
|
# NVidia Nsight GPU debugger configuration file
|
||||||
|
*.nvuser
|
||||||
|
|
||||||
|
# MFractors (Xamarin productivity tool) working folder
|
||||||
|
.mfractor/
|
||||||
|
|
||||||
|
# Local History for Visual Studio
|
||||||
|
.localhistory/
|
||||||
36
Blog.Server/Blog.Server.csproj
Normal file
36
Blog.Server/Blog.Server.csproj
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\NaiveHttpServer\NaiveHttpServer.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Handlebars.Net" Version="2.1.4" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<None Update="Views\index.html">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
|
<None Update="Static\css\feed.css">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
|
<None Update="Static\css\main.css">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
|
<None Update="Static\css\post.css">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
|
<None Update="Static\css\site.css">
|
||||||
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
|
</None>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
37
Blog.Server/Program.cs
Normal file
37
Blog.Server/Program.cs
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
// See https://aka.ms/new-console-template for more information
|
||||||
|
|
||||||
|
using HandlebarsDotNet;
|
||||||
|
using NaiveHttpServer;
|
||||||
|
|
||||||
|
|
||||||
|
var server = new Server("localhost", 3000);
|
||||||
|
|
||||||
|
var source = File.ReadAllText(Path.Combine(Environment.CurrentDirectory, "Views/index.html"));
|
||||||
|
|
||||||
|
var template = Handlebars.Compile(source);
|
||||||
|
|
||||||
|
var data = new {
|
||||||
|
title = "My new post",
|
||||||
|
body = "This is my first post!"
|
||||||
|
};
|
||||||
|
|
||||||
|
var result = template(data);
|
||||||
|
|
||||||
|
// Build Routers
|
||||||
|
var router = new RouterBuilder()
|
||||||
|
.Get("/", async ctx =>
|
||||||
|
{
|
||||||
|
await ctx.Response.Html(result);
|
||||||
|
}).Build();
|
||||||
|
server
|
||||||
|
.Use(Middlewares.Log)
|
||||||
|
.Use(Middlewares.ExceptionHandling)
|
||||||
|
.Use(Middlewares.StaticFile("css/", Path.Combine(Environment.CurrentDirectory,"Static/css/")))
|
||||||
|
.Use(router)
|
||||||
|
.Use(Middlewares.NotFound(documentUrl: "http://api.project.com/v1"));
|
||||||
|
|
||||||
|
server.Start();
|
||||||
|
|
||||||
|
Console.ReadKey();
|
||||||
|
|
||||||
|
server.Stop();
|
||||||
74
Blog.Server/Static/css/feed.css
Normal file
74
Blog.Server/Static/css/feed.css
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
.blog-title {
|
||||||
|
font-family: 'Roboto', serif;
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 300;
|
||||||
|
font-size: 32px;
|
||||||
|
line-height: 40px;
|
||||||
|
color: var(--bs-info);
|
||||||
|
}
|
||||||
|
|
||||||
|
.blog-nav-list {
|
||||||
|
font-weight: 400;
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 24px;
|
||||||
|
list-style: none;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.blog-nav-list > li {
|
||||||
|
float: left;
|
||||||
|
padding-right: 0.25rem;
|
||||||
|
color: #D9D9D9;
|
||||||
|
}
|
||||||
|
|
||||||
|
.active-nav-item {
|
||||||
|
color: var(--bs-info) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.box {
|
||||||
|
box-sizing: border-box;
|
||||||
|
background-clip: padding-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.blog-author-avatar {
|
||||||
|
max-height: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.blog-author {
|
||||||
|
size: 14px;
|
||||||
|
line-height: 32px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.post-link {
|
||||||
|
color: #4f4f4f !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tag {
|
||||||
|
border: 1px solid var(--bs-info);
|
||||||
|
border-radius: 4px;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 2px 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.static-avatar {
|
||||||
|
display: block;
|
||||||
|
overflow: hidden;
|
||||||
|
margin: 0 0 0 -6px;
|
||||||
|
width: 34px;
|
||||||
|
height: 34px;
|
||||||
|
border: 2px solid #fff;
|
||||||
|
border-radius: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.author-profile-image, .avatar-wrapper {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: #e3e9ed;
|
||||||
|
border-radius: 100%;
|
||||||
|
-o-object-fit: cover;
|
||||||
|
object-fit: cover;
|
||||||
|
}
|
||||||
205
Blog.Server/Static/css/main.css
Normal file
205
Blog.Server/Static/css/main.css
Normal file
@@ -0,0 +1,205 @@
|
|||||||
|
.timeline-card {
|
||||||
|
position: relative;
|
||||||
|
margin-left: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-card:before {
|
||||||
|
content: '';
|
||||||
|
display: inline-block;
|
||||||
|
position: absolute;
|
||||||
|
background-color: #fff;
|
||||||
|
border-radius: 100%;
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
top: 16px;
|
||||||
|
left: -12px;
|
||||||
|
border: 5px solid;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-body {
|
||||||
|
border-left: 2px solid #E6E9ED;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-card-primary:before {
|
||||||
|
border-color: var(--bs-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-card-info:before {
|
||||||
|
border-color: var(--bs-info);
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-card-secondary:before {
|
||||||
|
border-color: var(--bs-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.timeline-card-success:before {
|
||||||
|
border-color: var(--bs-teal);
|
||||||
|
}
|
||||||
|
|
||||||
|
html {
|
||||||
|
scroll-behavior: smooth;
|
||||||
|
}
|
||||||
|
|
||||||
|
html,
|
||||||
|
body {
|
||||||
|
overflow-x: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
max-width: 1140px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.site-title {
|
||||||
|
font-size: 1.25rem;
|
||||||
|
line-height: 2.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-link {
|
||||||
|
padding: 0;
|
||||||
|
font-size: 1rem;
|
||||||
|
line-height: 2.5rem;
|
||||||
|
color: inherit;
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-social .nav-link {
|
||||||
|
font-size: 1.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-link:hover,
|
||||||
|
.nav-link:focus {
|
||||||
|
color: inherit;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item+.nav-item {
|
||||||
|
margin-left: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cover {
|
||||||
|
height: 500px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cover>img {
|
||||||
|
-o-object-fit: cover;
|
||||||
|
object-fit: cover;
|
||||||
|
width: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress-bar {
|
||||||
|
text-transform: uppercase;
|
||||||
|
font-size: 10px;
|
||||||
|
letter-spacing: 1px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.text-small {
|
||||||
|
font-size: 0.85rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.text-teal {
|
||||||
|
color: var(--bs-teal);
|
||||||
|
}
|
||||||
|
|
||||||
|
footer a:not(.nav-link) {
|
||||||
|
color: inherit;
|
||||||
|
border-bottom: 1px dashed;
|
||||||
|
text-decoration: none;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.rounded-block {
|
||||||
|
border-radius: 0.5rem;
|
||||||
|
border: 1px solid rgba(0,0,0,.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 48em) {
|
||||||
|
.site-title {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.site-nav {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 767px) {
|
||||||
|
|
||||||
|
/* disable animations on mobile */
|
||||||
|
[data-aos] {
|
||||||
|
opacity: 1 !important;
|
||||||
|
transform: translate(0) scale(1) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-5 {
|
||||||
|
padding: 2.5rem 2rem !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.portfolio-section .m-5 {
|
||||||
|
margin: 2rem 0 1rem !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.portfolio-reverse {
|
||||||
|
flex-direction: column-reverse;
|
||||||
|
}
|
||||||
|
|
||||||
|
.portfolio-reverse .text-end {
|
||||||
|
text-align: start !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media print {
|
||||||
|
[data-aos] {
|
||||||
|
opacity: 1 !important;
|
||||||
|
transform: translate(0) scale(1) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
body.bg-light {
|
||||||
|
background-color: #fff !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container {
|
||||||
|
width: auto;
|
||||||
|
max-width: 100%;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cover {
|
||||||
|
height: 360px;
|
||||||
|
margin-bottom: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.cover>img {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shadow-1-strong {
|
||||||
|
box-shadow: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.resume-container>.my-5 {
|
||||||
|
margin: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.my-5.p-5 {
|
||||||
|
padding: 1.5rem 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.about-section {
|
||||||
|
padding: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.skills-section,
|
||||||
|
.work-experience-section,
|
||||||
|
.education-section,
|
||||||
|
.portfolio-section,
|
||||||
|
.reference-section,
|
||||||
|
.contact-section {
|
||||||
|
padding: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-break {
|
||||||
|
padding-top: 5rem;
|
||||||
|
page-break-before: always;
|
||||||
|
}
|
||||||
|
}
|
||||||
51
Blog.Server/Static/css/post.css
Normal file
51
Blog.Server/Static/css/post.css
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
.blog-title {
|
||||||
|
font-family: 'Roboto', serif;
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 300;
|
||||||
|
font-size: 32px;
|
||||||
|
line-height: 40px;
|
||||||
|
color: var(--bs-info);
|
||||||
|
}
|
||||||
|
|
||||||
|
.static-avatar {
|
||||||
|
display: block;
|
||||||
|
overflow: hidden;
|
||||||
|
margin: 0 0 0 -6px;
|
||||||
|
width: 40px;
|
||||||
|
height: 40px;
|
||||||
|
border: 2px solid #fff;
|
||||||
|
border-radius: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.author-profile-image, .avatar-wrapper {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: #e3e9ed;
|
||||||
|
border-radius: 100%;
|
||||||
|
-o-object-fit: cover;
|
||||||
|
object-fit: cover;
|
||||||
|
}
|
||||||
|
|
||||||
|
.post-author {
|
||||||
|
color: #757575;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wp-block-image > img {
|
||||||
|
max-width: 100%;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
figcaption {
|
||||||
|
color: #757575;
|
||||||
|
font-size: 75%;
|
||||||
|
line-height: 1.5em;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 768px) {
|
||||||
|
.post-content {
|
||||||
|
padding: 0 170px 6vw;
|
||||||
|
}
|
||||||
|
}
|
||||||
18
Blog.Server/Static/css/site.css
Normal file
18
Blog.Server/Static/css/site.css
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
html {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 768px) {
|
||||||
|
html {
|
||||||
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
html {
|
||||||
|
position: relative;
|
||||||
|
min-height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
margin-bottom: 60px;
|
||||||
|
}
|
||||||
110
Blog.Server/Views/index.html
Normal file
110
Blog.Server/Views/index.html
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
<meta name="description" content="Ivan Laletin - Senior Fullstack Developer's website and blog">
|
||||||
|
<meta property="og:title" content="Ivan Laletin - Senior Fullstack Developer" />
|
||||||
|
<meta property="og:description" content="Ivan Laletin - Senior Fullstack Developer's website and blog">
|
||||||
|
<meta property="og:site_name" content="Ivan Laletin">
|
||||||
|
<meta property="og:type" content="profile" />
|
||||||
|
<meta property="og:image" content="/images/I01P00012-min.webp" />
|
||||||
|
<meta property="og:url" content="https://e1lama.ru" />
|
||||||
|
<link rel="canonical" href="https://e1lama.ru">
|
||||||
|
<meta property="og:locale" content="en_US">
|
||||||
|
<meta property="profile:first_name" content="Ivan">
|
||||||
|
<meta property="profile:last_name" content="Laletin">
|
||||||
|
<meta property="profile:gender" content="male">
|
||||||
|
<meta property="profile:username" content="e1lama">
|
||||||
|
<title>Ivan Laletin - Senior Fullstack Developer</title>
|
||||||
|
<link rel="icon" type="image/x-icon" href="~/favicon.ico">
|
||||||
|
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin="crossorigin" />
|
||||||
|
<link rel="preload" as="style"
|
||||||
|
href="https://fonts.googleapis.com/css2?family=Poppins:wght@600&family=Roboto:wght@300;400;500;700&display=swap" />
|
||||||
|
<link rel="stylesheet"
|
||||||
|
href="https://fonts.googleapis.com/css2?family=Poppins:wght@600&family=Roboto:wght@300;400;500;700&display=swap"
|
||||||
|
media="print" onload="this.media='all'" />
|
||||||
|
<noscript>
|
||||||
|
<link rel="stylesheet"
|
||||||
|
href="https://fonts.googleapis.com/css2?family=Poppins:wght@600&family=Roboto:wght@300;400;500;700&display=swap" />
|
||||||
|
</noscript>
|
||||||
|
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/5.15.1/css/all.min.css" integrity="sha512-+4zCK9k+qNFUR5X+cKL9EIR+ZOhtIloNl9GIKS57V1MyNsYpYcUrUeQc9vNfzsWfV28IaLL3i96P9sdNyeRssA==" crossorigin="anonymous" referrerpolicy="no-referrer" />
|
||||||
|
<link href="https://cdnjs.cloudflare.com/ajax/libs/mdb-ui-kit/3.2.0/mdb.min.css" rel="stylesheet" media="all">
|
||||||
|
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/aos/2.3.4/aos.css" integrity="sha512-1cK78a1o+ht2JcaW6g8OXYwqpev9+6GqOkz9xmBN9iUUhIndKtxwILGWYOSibOKjLsEdjyjZvYDq/cZwNeak0w==" crossorigin="anonymous" referrerpolicy="no-referrer" />
|
||||||
|
<link href="/css/main.css" rel="stylesheet">
|
||||||
|
<noscript>
|
||||||
|
<style type="text/css">
|
||||||
|
[data-aos] {
|
||||||
|
opacity: 1 !important;
|
||||||
|
transform: translate(0) scale(1) !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</noscript>
|
||||||
|
</head>
|
||||||
|
<body class="bg-light" id="top">
|
||||||
|
<header class="d-print-none">
|
||||||
|
<div class="container text-center text-lg-left">
|
||||||
|
<div class="pt-4 clearfix">
|
||||||
|
<h1 class="site-title mb-0">Ivan Laletin</h1>
|
||||||
|
<div class="site-nav">
|
||||||
|
<nav role="navigation">
|
||||||
|
<ul class="nav justify-content-center">
|
||||||
|
<li class="nav-item">
|
||||||
|
<a class="nav-link" title="Home" asp-area="" asp-page="/Index">
|
||||||
|
<span
|
||||||
|
class="menu-title">
|
||||||
|
Home
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li class="nav-item">
|
||||||
|
<a class="nav-link" title="Blog" asp-area="" asp-page="/Feed">
|
||||||
|
<span
|
||||||
|
class="menu-title">
|
||||||
|
Blog
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
<main class="page-content">
|
||||||
|
{{body}}
|
||||||
|
</main>
|
||||||
|
|
||||||
|
<footer class="pt-4 pb-4 text-muted text-center d-print-none">
|
||||||
|
<div class="container">
|
||||||
|
<div class="my-3">
|
||||||
|
<div class="h4">Ivan Laletin</div>
|
||||||
|
<div class="footer-nav">
|
||||||
|
<nav role="navigation">
|
||||||
|
<ul class="nav justify-content-center">
|
||||||
|
<li class="nav-item"><a class="nav-link" href="https://twitter.com/backndr" title="Twitter"><i
|
||||||
|
class="fab fa-twitter"></i><span class="menu-title sr-only">Twitter</span></a>
|
||||||
|
</li>
|
||||||
|
<li class="nav-item"><a class="nav-link" href="https://github.com/HiveBeats" title="Github"><i
|
||||||
|
class="fab fa-github"></i><span class="menu-title sr-only">Github</span></a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="text-small">
|
||||||
|
<!-- todo: add, now is missed -->
|
||||||
|
<a href="./ivan.asc">PGP Public Key</a> Fingerprint: E4B4 1292 8F1A B309 7BB1 0D41 2F55 0236 0018 36A7
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- Please feel free to buy me a coffee -->
|
||||||
|
<!-- BTC: 1Nvb7A45ZGmS5zSSyDWWDFV7CnCYGKSUPV -->
|
||||||
|
<!-- TON: EQAmDR5k0t-SBULTN4cv63MHweHuD-r3FNuvshfe86L4cA5M -->
|
||||||
|
<!-- USDT (TRC20): TFPRYNgR5gLqE8WaLeDAf1RpWux2hdNdbK -->
|
||||||
|
</footer>
|
||||||
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/mdb-ui-kit/3.2.0/mdb.min.js"></script>
|
||||||
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/aos/2.3.4/aos.js" integrity="sha512-A7AYk1fGKX6S2SsHywmPkrnzTZHrgiVT7GcQkLGDe2ev0aWb8zejytzS8wjo7PGEXKqJOrjQ4oORtnimIRZBtw==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
|
||||||
|
<!--<script src="/js/main.js?ver=1.2.1"></script>-->
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
14
Demo/Demo.csproj
Normal file
14
Demo/Demo.csproj
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\NaiveHttpServer\NaiveHttpServer.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
15
Demo/Program.cs
Normal file
15
Demo/Program.cs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
using NaiveHttpServer;
|
||||||
|
|
||||||
|
var server = new Server("localhost", 2333);
|
||||||
|
|
||||||
|
server
|
||||||
|
.Use(Middlewares.Log)
|
||||||
|
.Use(Middlewares.ExceptionHandling)
|
||||||
|
.Use(Middlewares.StaticFile("/files", Environment.CurrentDirectory))
|
||||||
|
.Use(Middlewares.NotFound(documentUrl: "http://api.project.com/v1"));
|
||||||
|
|
||||||
|
server.Start();
|
||||||
|
|
||||||
|
Console.ReadKey();
|
||||||
|
|
||||||
|
server.Stop();
|
||||||
21
LICENSE
Normal file
21
LICENSE
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2021 Dingping Zhang
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
||||||
45
NaiveHttpServer.sln
Normal file
45
NaiveHttpServer.sln
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
|
||||||
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
|
# Visual Studio Version 17
|
||||||
|
VisualStudioVersion = 17.3.32929.385
|
||||||
|
MinimumVisualStudioVersion = 10.0.40219.1
|
||||||
|
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "NaiveHttpServer", "NaiveHttpServer\NaiveHttpServer.csproj", "{DBE214C2-832D-4F3A-8AA5-1B3717EFEE2A}"
|
||||||
|
EndProject
|
||||||
|
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{F9C0C1FB-0DBA-444E-A01C-A2BD05127557}"
|
||||||
|
ProjectSection(SolutionItems) = preProject
|
||||||
|
.editorconfig = .editorconfig
|
||||||
|
.gitignore = .gitignore
|
||||||
|
LICENSE = LICENSE
|
||||||
|
README.md = README.md
|
||||||
|
EndProjectSection
|
||||||
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Demo", "Demo\Demo.csproj", "{173B9B8A-AD3D-4BC3-917D-9E56E7B13681}"
|
||||||
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Blog.Server", "Blog.Server\Blog.Server.csproj", "{1137F0C8-0B4A-4C6B-9FAD-4481E0E2C588}"
|
||||||
|
EndProject
|
||||||
|
Global
|
||||||
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
|
Debug|Any CPU = Debug|Any CPU
|
||||||
|
Release|Any CPU = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||||
|
{DBE214C2-832D-4F3A-8AA5-1B3717EFEE2A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{DBE214C2-832D-4F3A-8AA5-1B3717EFEE2A}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{DBE214C2-832D-4F3A-8AA5-1B3717EFEE2A}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{DBE214C2-832D-4F3A-8AA5-1B3717EFEE2A}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{173B9B8A-AD3D-4BC3-917D-9E56E7B13681}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{173B9B8A-AD3D-4BC3-917D-9E56E7B13681}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{173B9B8A-AD3D-4BC3-917D-9E56E7B13681}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{173B9B8A-AD3D-4BC3-917D-9E56E7B13681}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{1137F0C8-0B4A-4C6B-9FAD-4481E0E2C588}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{1137F0C8-0B4A-4C6B-9FAD-4481E0E2C588}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{1137F0C8-0B4A-4C6B-9FAD-4481E0E2C588}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{1137F0C8-0B4A-4C6B-9FAD-4481E0E2C588}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
|
HideSolutionNode = FALSE
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||||
|
SolutionGuid = {5472BDAB-9DDB-46B0-996F-B1FE7DF2035C}
|
||||||
|
EndGlobalSection
|
||||||
|
EndGlobal
|
||||||
29
NaiveHttpServer/Context.cs
Normal file
29
NaiveHttpServer/Context.cs
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
using System.Net;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public delegate bool ParameterProvider(string key, out string value);
|
||||||
|
|
||||||
|
public class Context
|
||||||
|
{
|
||||||
|
public HttpListenerRequest Request { get; }
|
||||||
|
|
||||||
|
public HttpListenerResponse Response { get; }
|
||||||
|
|
||||||
|
public ILogger Logger { get; }
|
||||||
|
|
||||||
|
public ParameterProvider TryGetParameter { get; set; }
|
||||||
|
|
||||||
|
public Context(HttpListenerRequest request, HttpListenerResponse response, ILogger logger)
|
||||||
|
{
|
||||||
|
Request = request;
|
||||||
|
Response = response;
|
||||||
|
Logger = logger;
|
||||||
|
TryGetParameter = (string _, out string value) =>
|
||||||
|
{
|
||||||
|
value = null!;
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
38
NaiveHttpServer/DefaultLogger.cs
Normal file
38
NaiveHttpServer/DefaultLogger.cs
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
internal class DefaultLogger : ILogger
|
||||||
|
{
|
||||||
|
public void Error(string message, Exception? exception = null)
|
||||||
|
{
|
||||||
|
Write(nameof(Error), message, exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Warning(string message, Exception? exception = null)
|
||||||
|
{
|
||||||
|
Write(nameof(Warning), message, exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Info(string message, Exception? exception = null)
|
||||||
|
{
|
||||||
|
Write(nameof(Info), message, exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Debug(string message, Exception? exception = null)
|
||||||
|
{
|
||||||
|
Write(nameof(Debug), message, exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Write(string level, string message, Exception? exception)
|
||||||
|
{
|
||||||
|
Thread thread = Thread.CurrentThread;
|
||||||
|
string threadName = string.IsNullOrEmpty(thread.Name) ? thread.ManagedThreadId.ToString() : thread.Name!;
|
||||||
|
|
||||||
|
string exceptionString = exception is null ? string.Empty : $"{Environment.NewLine}{exception.Message}{Environment.NewLine}{exception.StackTrace}{Environment.NewLine}";
|
||||||
|
|
||||||
|
System.Diagnostics.Debug.WriteLine($"{DateTime.Now:HH:mm:ss.fff} {level.ToUpperInvariant()} [{threadName}] {message}{exceptionString}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
9
NaiveHttpServer/ErrorCodes.cs
Normal file
9
NaiveHttpServer/ErrorCodes.cs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public static class ErrorCodes
|
||||||
|
{
|
||||||
|
public const string Unknown = "UNKNOWN";
|
||||||
|
public const string NotFoundApi = "NOT_FOUND_API";
|
||||||
|
public const string NotFoundFile = "NOT_FOUND_FILE";
|
||||||
|
}
|
||||||
|
}
|
||||||
83
NaiveHttpServer/Extensions.cs
Normal file
83
NaiveHttpServer/Extensions.cs
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
using System.IO;
|
||||||
|
using System.Net;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Newtonsoft.Json;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public static class Extensions
|
||||||
|
{
|
||||||
|
public static readonly JsonSerializerSettings JsonSettings = new()
|
||||||
|
{
|
||||||
|
MissingMemberHandling = MissingMemberHandling.Ignore,
|
||||||
|
NullValueHandling = NullValueHandling.Ignore,
|
||||||
|
};
|
||||||
|
|
||||||
|
public static async Task Json(this HttpListenerResponse response, object value)
|
||||||
|
{
|
||||||
|
string jsonText = value.ToJson();
|
||||||
|
byte[] bytes = Encoding.UTF8.GetBytes(jsonText);
|
||||||
|
|
||||||
|
response.ContentEncoding = Encoding.UTF8;
|
||||||
|
response.ContentType = "application/json";
|
||||||
|
response.ContentLength64 = bytes.Length;
|
||||||
|
await response.OutputStream.WriteAsync(bytes, 0, bytes.Length);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task File(this HttpListenerResponse response, string filePath)
|
||||||
|
{
|
||||||
|
if (!System.IO.File.Exists(filePath))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await FileHelper.ReadAsync(filePath, async stream =>
|
||||||
|
{
|
||||||
|
response.ContentType = MimeTypes.GetMimeType(filePath);
|
||||||
|
response.ContentLength64 = stream.Length;
|
||||||
|
await stream.CopyToAsync(response.OutputStream);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task Html(this HttpListenerResponse response, string template)
|
||||||
|
{
|
||||||
|
byte[] buffer = Encoding.UTF8.GetBytes(template);
|
||||||
|
response.ContentType = System.Net.Mime.MediaTypeNames.Text.Html;
|
||||||
|
response.ContentLength64 = buffer.Length;
|
||||||
|
await response.OutputStream.WriteAsync(buffer, 0, buffer.Length);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task Error(this HttpListenerResponse response, string errorCode, string message, int statusCode = 500)
|
||||||
|
{
|
||||||
|
response.StatusCode = statusCode;
|
||||||
|
await response.Json(new
|
||||||
|
{
|
||||||
|
errorCode,
|
||||||
|
message,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task<T?> JsonFromBody<T>(this HttpListenerRequest request)
|
||||||
|
{
|
||||||
|
string jsonText = await request.TextFromBody();
|
||||||
|
return jsonText.ToObject<T>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task<string> TextFromBody(this HttpListenerRequest request)
|
||||||
|
{
|
||||||
|
using StreamReader reader = new(request.InputStream);
|
||||||
|
return await reader.ReadToEndAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string ToJson(this object value, Formatting formatting = Formatting.None)
|
||||||
|
{
|
||||||
|
return JsonConvert.SerializeObject(value, formatting, JsonSettings);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static T? ToObject<T>(this string json)
|
||||||
|
{
|
||||||
|
return JsonConvert.DeserializeObject<T>(json, JsonSettings);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
89
NaiveHttpServer/FileHelper.cs
Normal file
89
NaiveHttpServer/FileHelper.cs
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
using System;
|
||||||
|
using System.IO;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public static class FileHelper
|
||||||
|
{
|
||||||
|
public static ILogger? Logger { get; set; }
|
||||||
|
|
||||||
|
public static async Task WriteAsync(string path, Func<Stream, Task> writer, bool isBackup = true)
|
||||||
|
{
|
||||||
|
string tempFilePath = $"{path}.writing";
|
||||||
|
using (var stream = new FileStream(
|
||||||
|
tempFilePath,
|
||||||
|
FileMode.Create,
|
||||||
|
FileAccess.Write,
|
||||||
|
FileShare.None,
|
||||||
|
0x10000,
|
||||||
|
FileOptions.SequentialScan)
|
||||||
|
)
|
||||||
|
{
|
||||||
|
await writer(stream);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (File.Exists(path))
|
||||||
|
{
|
||||||
|
await SpinRetry(() => File.Replace(tempFilePath, path, isBackup ? BackupPath(path) : null, true));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await SpinRetry(() => File.Move(tempFilePath, path));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task ReadAsync(string path, Func<Stream, Task> reader)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await CriticalReadAsync(path, reader);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
string backupPath = BackupPath(path);
|
||||||
|
if (!File.Exists(backupPath))
|
||||||
|
{
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
|
||||||
|
Logger?.Warning($"Can not read {path}, turn back to backup.", e);
|
||||||
|
await CriticalReadAsync(backupPath, reader);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task CriticalReadAsync(string path, Func<Stream, Task> reader)
|
||||||
|
{
|
||||||
|
using FileStream stream = new(
|
||||||
|
path,
|
||||||
|
FileMode.OpenOrCreate,
|
||||||
|
FileAccess.Read,
|
||||||
|
FileShare.ReadWrite,
|
||||||
|
0x10000,
|
||||||
|
FileOptions.SequentialScan);
|
||||||
|
await reader(stream);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string BackupPath(string path) => $"{path}.backup";
|
||||||
|
|
||||||
|
private static async Task SpinRetry(Action action, int retryCount = 10)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < retryCount; i++)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
action();
|
||||||
|
await Task.Delay(100);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
if (i == retryCount - 1)
|
||||||
|
{
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
10
NaiveHttpServer/HttpMethods.cs
Normal file
10
NaiveHttpServer/HttpMethods.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public static class HttpMethods
|
||||||
|
{
|
||||||
|
public const string Get = "GET";
|
||||||
|
public const string Post = "POST";
|
||||||
|
public const string Put = "PUT";
|
||||||
|
public const string Delete = "DELETE";
|
||||||
|
}
|
||||||
|
}
|
||||||
15
NaiveHttpServer/ILogger.cs
Normal file
15
NaiveHttpServer/ILogger.cs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
using System;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public interface ILogger
|
||||||
|
{
|
||||||
|
void Error(string message, Exception? exception = null);
|
||||||
|
|
||||||
|
void Warning(string message, Exception? exception = null);
|
||||||
|
|
||||||
|
void Info(string message, Exception? exception = null);
|
||||||
|
|
||||||
|
void Debug(string message, Exception? exception = null);
|
||||||
|
}
|
||||||
|
}
|
||||||
18
NaiveHttpServer/IRouterBuilder.cs
Normal file
18
NaiveHttpServer/IRouterBuilder.cs
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
using System;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public interface IRouterBuilder
|
||||||
|
{
|
||||||
|
IRouterBuilder Get(string url, Func<Context, Task> handler);
|
||||||
|
|
||||||
|
IRouterBuilder Post(string url, Func<Context, Task> handler);
|
||||||
|
|
||||||
|
IRouterBuilder Delete(string url, Func<Context, Task> handler);
|
||||||
|
|
||||||
|
IRouterBuilder Put(string url, Func<Context, Task> handler);
|
||||||
|
|
||||||
|
Middleware<Context> Build();
|
||||||
|
}
|
||||||
|
}
|
||||||
134
NaiveHttpServer/Middlewares.cs
Normal file
134
NaiveHttpServer/Middlewares.cs
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
using System;
|
||||||
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Net;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Web;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public delegate Task Middleware<in T>(T ctx, Func<Task> next);
|
||||||
|
|
||||||
|
public static class Middlewares
|
||||||
|
{
|
||||||
|
public static Middleware<T> Empty<T>() => (_, next) => next();
|
||||||
|
|
||||||
|
public static async Task Log(Context ctx, Func<Task> next)
|
||||||
|
{
|
||||||
|
ILogger logger = ctx.Logger;
|
||||||
|
HttpListenerRequest request = ctx.Request;
|
||||||
|
HttpListenerResponse response = ctx.Response;
|
||||||
|
|
||||||
|
logger.Debug($"[In] {request.HttpMethod} {request.RawUrl}");
|
||||||
|
await next();
|
||||||
|
logger.Debug($"[Out] {request.HttpMethod} [{response.StatusCode}] {request.RawUrl}");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task ExceptionHandling(Context ctx, Func<Task> next)
|
||||||
|
{
|
||||||
|
using HttpListenerResponse response = ctx.Response;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await next();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
ctx.Logger.Warning("Unexpected exception occurred.", e);
|
||||||
|
await response.Error(ErrorCodes.Unknown, e.Message);
|
||||||
|
response.StatusCode = e switch
|
||||||
|
{
|
||||||
|
FileNotFoundException => 404,
|
||||||
|
DirectoryNotFoundException => 404,
|
||||||
|
UnauthorizedAccessException => 403,
|
||||||
|
_ => 500,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Middleware<Context> NotFound(string documentUrl)
|
||||||
|
{
|
||||||
|
return async (ctx, _) =>
|
||||||
|
{
|
||||||
|
ctx.Response.StatusCode = 404;
|
||||||
|
await ctx.Response.Error(
|
||||||
|
ErrorCodes.NotFoundApi,
|
||||||
|
$"Not found this api: '{ctx.Request.RawUrl}', and please read the API document: {documentUrl}.");
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Middleware<Context> StaticFile(string route, string rootDir)
|
||||||
|
{
|
||||||
|
return async (ctx, next) =>
|
||||||
|
{
|
||||||
|
// Don't use Request.RawUrl, because it contains url parameters. (e.g. '?a=1&b=2')
|
||||||
|
string relativePath = ctx.Request.Url.AbsolutePath.TrimStart('/');
|
||||||
|
bool handled = relativePath.StartsWith(route);
|
||||||
|
if (!handled)
|
||||||
|
{
|
||||||
|
await next();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
string requestPath = HttpUtility.UrlDecode(relativePath)
|
||||||
|
.Substring(route.Length)
|
||||||
|
.ToLowerInvariant()
|
||||||
|
.TrimStart('/', '\\');
|
||||||
|
string filePath = Path.Combine(rootDir, requestPath);
|
||||||
|
|
||||||
|
switch (ctx.Request.HttpMethod)
|
||||||
|
{
|
||||||
|
case HttpMethods.Get:
|
||||||
|
await ReadLocalFile(filePath, ctx.Response, ctx.Logger);
|
||||||
|
break;
|
||||||
|
case HttpMethods.Put:
|
||||||
|
await WriteLocalFile(filePath, ctx.Request);
|
||||||
|
ctx.Response.StatusCode = 204;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task ReadLocalFile(string filePath, HttpListenerResponse response, ILogger logger)
|
||||||
|
{
|
||||||
|
if (File.Exists(filePath))
|
||||||
|
{
|
||||||
|
await response.File(filePath);
|
||||||
|
}
|
||||||
|
else if (Directory.Exists(filePath))
|
||||||
|
{
|
||||||
|
string[] filePaths = Directory
|
||||||
|
.GetFileSystemEntries(filePath)
|
||||||
|
.Select(Path.GetFileName)
|
||||||
|
.ToArray();
|
||||||
|
await response.Json(filePaths);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
string message = $"Not found the file: '{filePath}'.";
|
||||||
|
logger.Warning(message);
|
||||||
|
await response.Error(ErrorCodes.NotFoundFile, message, 404);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Task WriteLocalFile(string filePath, HttpListenerRequest request)
|
||||||
|
{
|
||||||
|
return FileHelper.WriteAsync(filePath, stream => request.InputStream.CopyToAsync(stream));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Middleware<T> Then<T>(this Middleware<T> middleware, Middleware<T> nextMiddleware)
|
||||||
|
{
|
||||||
|
return (ctx, next) => middleware(ctx, () => nextMiddleware(ctx, next));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Task Run<T>(this Middleware<T> middleware, T ctx)
|
||||||
|
{
|
||||||
|
return middleware(ctx, () =>
|
||||||
|
#if NET45
|
||||||
|
Task.FromResult(0)
|
||||||
|
#else
|
||||||
|
Task.CompletedTask
|
||||||
|
#endif
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
1032
NaiveHttpServer/MimeTypes.cs
Normal file
1032
NaiveHttpServer/MimeTypes.cs
Normal file
File diff suppressed because it is too large
Load Diff
95
NaiveHttpServer/MimeTypes.tt
Normal file
95
NaiveHttpServer/MimeTypes.tt
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
<#@ template debug="false" hostspecific="false" language="C#" #>
|
||||||
|
<#@ assembly name="System.Core" #>
|
||||||
|
<#@ import namespace="System.Linq" #>
|
||||||
|
<#@ import namespace="System.Text" #>
|
||||||
|
<#@ import namespace="System.Net" #>
|
||||||
|
<#@ import namespace="System.Collections.Generic" #>
|
||||||
|
<#@ output extension=".cs" #>
|
||||||
|
<# var mediaTypes = GetMediaTypeList(); #>
|
||||||
|
// <auto-generated />
|
||||||
|
|
||||||
|
namespace <#=System.Runtime.Remoting.Messaging.CallContext.LogicalGetData("NamespaceHint").ToString()#>
|
||||||
|
{
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Provides utilities for mapping file names and extensions to MIME-types.
|
||||||
|
/// </summary>
|
||||||
|
[CompilerGenerated]
|
||||||
|
[DebuggerNonUserCode]
|
||||||
|
public static class MimeTypes
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// The fallback MIME-type. Defaults to <c>application/octet-stream</c>.
|
||||||
|
/// </summary>
|
||||||
|
public static string FallbackMimeType { get; set; }
|
||||||
|
|
||||||
|
private static readonly Dictionary<string, string> TypeMap;
|
||||||
|
|
||||||
|
static MimeTypes()
|
||||||
|
{
|
||||||
|
FallbackMimeType = "application/octet-stream";
|
||||||
|
|
||||||
|
TypeMap = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase)
|
||||||
|
{
|
||||||
|
<# foreach (var mediaType in mediaTypes) { #>
|
||||||
|
{ "<#= mediaType.Item1 #>", "<#= mediaType.Item2 #>" },
|
||||||
|
<# } #>
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the MIME-type for the given file name,
|
||||||
|
/// or <see cref="FallbackMimeType"/> if a mapping doesn't exist.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="fileName">The name of the file.</param>
|
||||||
|
/// <returns>The MIME-type for the given file name.</returns>
|
||||||
|
public static string GetMimeType(string fileName)
|
||||||
|
{
|
||||||
|
var dotIndex = fileName.LastIndexOf('.');
|
||||||
|
|
||||||
|
return dotIndex != -1 &&
|
||||||
|
fileName.Length > dotIndex + 1 &&
|
||||||
|
TypeMap.TryGetValue(fileName.Substring(dotIndex + 1), out var result)
|
||||||
|
? result
|
||||||
|
: FallbackMimeType;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<#+
|
||||||
|
private static IList<Tuple<string, string>> GetMediaTypeList()
|
||||||
|
{
|
||||||
|
using (var client = new WebClient())
|
||||||
|
{
|
||||||
|
var list = client.DownloadString(new Uri("http://svn.apache.org/repos/asf/httpd/httpd/trunk/docs/conf/mime.types"));
|
||||||
|
|
||||||
|
var lines = SplitString(list, '\r', '\n');
|
||||||
|
|
||||||
|
return GetMediaTypes(lines).ToList();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static IEnumerable<Tuple<string, string>> GetMediaTypes(IEnumerable<string> lines)
|
||||||
|
{
|
||||||
|
return lines.Where(x => !x.StartsWith("#"))
|
||||||
|
.Select(line => SplitString(line, '\t', ' '))
|
||||||
|
.SelectMany(CreateMediaTypes)
|
||||||
|
.GroupBy(x => x.Item1)
|
||||||
|
.Where(x => x.Count() == 1)
|
||||||
|
.Select(x => x.Single())
|
||||||
|
.OrderBy(x => x.Item1);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static string[] SplitString(string line, params char[] separator)
|
||||||
|
{
|
||||||
|
return line.Split(separator, StringSplitOptions.RemoveEmptyEntries);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static IEnumerable<Tuple<string, string>> CreateMediaTypes(string[] parts)
|
||||||
|
{
|
||||||
|
return parts.Skip(1).Select(extension => Tuple.Create(extension, parts.First()));
|
||||||
|
}
|
||||||
|
#>
|
||||||
50
NaiveHttpServer/NaiveHttpServer.csproj
Normal file
50
NaiveHttpServer/NaiveHttpServer.csproj
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFrameworks>netstandard2.0;net462</TargetFrameworks>
|
||||||
|
<Version>1.0.0</Version>
|
||||||
|
<LangVersion>latest</LangVersion>
|
||||||
|
<EnableNETAnalyzers>true</EnableNETAnalyzers>
|
||||||
|
<EnforceCodeStyleInBuild>true</EnforceCodeStyleInBuild>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
|
||||||
|
<GeneratePackageOnBuild Condition="'$(Configuration)'=='Release'">true</GeneratePackageOnBuild>
|
||||||
|
<Authors>Dingping Zhang</Authors>
|
||||||
|
<Copyright>Copyright (c) 2020-2021 Dingping Zhang</Copyright>
|
||||||
|
<PackageProjectUrl>https://github.com/DingpingZhang/NaiveHttpServer</PackageProjectUrl>
|
||||||
|
<RepositoryUrl>https://github.com/DingpingZhang/NaiveHttpServer</RepositoryUrl>
|
||||||
|
<PublishRepositoryUrl>true</PublishRepositoryUrl>
|
||||||
|
<PackageLicenseExpression>MIT</PackageLicenseExpression>
|
||||||
|
<Description>A simple C# http server based on the HttpListener.</Description>
|
||||||
|
<PackageTags>http;http-server;file-server;simple-http-server</PackageTags>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Newtonsoft.Json" Version="[13.0.1,)" />
|
||||||
|
<PackageReference Condition="'$(TargetFramework)' == 'net462'" Include="System.ValueTuple" Version="[4.5.0,)" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Reference Condition="'$(TargetFramework)' == 'net462'" Include="System.Web" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<None Update="MimeTypes.tt">
|
||||||
|
<Generator>TextTemplatingFileGenerator</Generator>
|
||||||
|
<LastGenOutput>MimeTypes.cs</LastGenOutput>
|
||||||
|
</None>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Service Include="{508349b6-6b84-4df5-91f0-309beebad82d}" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Compile Update="MimeTypes.cs">
|
||||||
|
<DesignTime>True</DesignTime>
|
||||||
|
<AutoGen>True</AutoGen>
|
||||||
|
<DependentUpon>MimeTypes.tt</DependentUpon>
|
||||||
|
</Compile>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
50
NaiveHttpServer/NetAclChecker.cs
Normal file
50
NaiveHttpServer/NetAclChecker.cs
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
using System;
|
||||||
|
using System.ComponentModel;
|
||||||
|
using System.Diagnostics;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public static class NetAclChecker
|
||||||
|
{
|
||||||
|
public static ILogger? Logger { get; set; }
|
||||||
|
|
||||||
|
public static void AddAddress(string address)
|
||||||
|
{
|
||||||
|
AddAddress(address, Environment.UserDomainName, Environment.UserName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void AddAddress(string address, string domain, string user)
|
||||||
|
{
|
||||||
|
string args = $@"http add urlacl url={address}, user={domain}\{user}";
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ProcessStartInfo processStartInfo = new("netsh", args)
|
||||||
|
{
|
||||||
|
Verb = "runas",
|
||||||
|
CreateNoWindow = true,
|
||||||
|
WindowStyle = ProcessWindowStyle.Hidden,
|
||||||
|
UseShellExecute = true,
|
||||||
|
};
|
||||||
|
|
||||||
|
var process = Process.Start(processStartInfo);
|
||||||
|
process?.WaitForExit();
|
||||||
|
}
|
||||||
|
catch (Win32Exception e)
|
||||||
|
{
|
||||||
|
if (e.NativeErrorCode == 1223)
|
||||||
|
{
|
||||||
|
Logger?.Info("User canceled the operation by rejected the UAC.");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Logger?.Warning($"Failed to 'netsh http add urlacl {address}' with an {nameof(Win32Exception)}.", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
Logger?.Warning($"Failed to 'netsh http add urlacl {address}'.", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
135
NaiveHttpServer/RouterBuilder.cs
Normal file
135
NaiveHttpServer/RouterBuilder.cs
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Collections.Specialized;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Web;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public class RouterBuilder : IRouterBuilder
|
||||||
|
{
|
||||||
|
private static readonly Regex PathParameterRegex = new("(?<=/):(.+?)(?:(?=/)|$)", RegexOptions.Compiled);
|
||||||
|
private static readonly char[] Separator = { '/' };
|
||||||
|
|
||||||
|
private readonly List<(string url, Func<Context, Task> handler)> _getRoutes = new();
|
||||||
|
private readonly List<(string url, Func<Context, Task> handler)> _postRoutes = new();
|
||||||
|
private readonly List<(string url, Func<Context, Task> handler)> _deleteRoutes = new();
|
||||||
|
private readonly List<(string url, Func<Context, Task> handler)> _putRoutes = new();
|
||||||
|
|
||||||
|
public IRouterBuilder Get(string url, Func<Context, Task> handler)
|
||||||
|
{
|
||||||
|
_getRoutes.Add((url, handler));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IRouterBuilder Post(string url, Func<Context, Task> handler)
|
||||||
|
{
|
||||||
|
_postRoutes.Add((url, handler));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IRouterBuilder Delete(string url, Func<Context, Task> handler)
|
||||||
|
{
|
||||||
|
_deleteRoutes.Add((url, handler));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IRouterBuilder Put(string url, Func<Context, Task> handler)
|
||||||
|
{
|
||||||
|
_putRoutes.Add((url, handler));
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Middleware<Context> Build()
|
||||||
|
{
|
||||||
|
var getRoutes = GenerateRegexRoutes(_getRoutes);
|
||||||
|
var postRoutes = GenerateRegexRoutes(_postRoutes);
|
||||||
|
var deleteRoutes = GenerateRegexRoutes(_deleteRoutes);
|
||||||
|
var putRoutes = GenerateRegexRoutes(_putRoutes);
|
||||||
|
|
||||||
|
return async (ctx, next) =>
|
||||||
|
{
|
||||||
|
bool handled = ctx.Request.HttpMethod.ToUpperInvariant() switch
|
||||||
|
{
|
||||||
|
HttpMethods.Get => await TryMatch(getRoutes, ctx),
|
||||||
|
HttpMethods.Post => await TryMatch(postRoutes, ctx),
|
||||||
|
HttpMethods.Delete => await TryMatch(deleteRoutes, ctx),
|
||||||
|
HttpMethods.Put => await TryMatch(putRoutes, ctx),
|
||||||
|
_ => false,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!handled)
|
||||||
|
{
|
||||||
|
await next();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static IReadOnlyList<(Regex regex, Func<Context, Task> handler)> GenerateRegexRoutes(IEnumerable<(string url, Func<Context, Task> handler)> routes)
|
||||||
|
{
|
||||||
|
var toSortRoutes = routes
|
||||||
|
.Select(item => (
|
||||||
|
fragmentCount: item.url.Split(Separator, StringSplitOptions.RemoveEmptyEntries).Length,
|
||||||
|
item.url,
|
||||||
|
item.handler))
|
||||||
|
.ToList();
|
||||||
|
toSortRoutes.Sort((x, y) => y.fragmentCount - x.fragmentCount);
|
||||||
|
|
||||||
|
return toSortRoutes
|
||||||
|
.Select(item => (regex: GetPathRegex(item.url), item.handler))
|
||||||
|
.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Regex GetPathRegex(string url)
|
||||||
|
{
|
||||||
|
HashSet<string> parameterNames = new();
|
||||||
|
string urlRegex = PathParameterRegex
|
||||||
|
.Replace(url.Trim('/'), match =>
|
||||||
|
{
|
||||||
|
if (!parameterNames.Add(match.Value))
|
||||||
|
{
|
||||||
|
throw new ArgumentException($"Cannot contains duplicate variable name: '{match.Value}'.", nameof(url));
|
||||||
|
}
|
||||||
|
|
||||||
|
return $"(?<{match.Groups[1]}>.+?)";
|
||||||
|
});
|
||||||
|
|
||||||
|
return new Regex($"{urlRegex}$", RegexOptions.Compiled);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<bool> TryMatch(IEnumerable<(Regex regex, Func<Context, Task> handler)> routes, Context ctx)
|
||||||
|
{
|
||||||
|
string requestPath = ctx.Request.Url.LocalPath.ToLowerInvariant();
|
||||||
|
|
||||||
|
foreach ((Regex regex, Func<Context, Task> handler) in routes)
|
||||||
|
{
|
||||||
|
Match match = regex.Match(requestPath);
|
||||||
|
if (match.Success)
|
||||||
|
{
|
||||||
|
NameValueCollection query = HttpUtility.ParseQueryString(ctx.Request.Url.Query, Encoding.UTF8);
|
||||||
|
|
||||||
|
ctx.TryGetParameter = (string key, out string value) =>
|
||||||
|
{
|
||||||
|
Group group = match.Groups[key];
|
||||||
|
value = HttpUtility.UrlDecode(group.Value);
|
||||||
|
|
||||||
|
if (!group.Success)
|
||||||
|
{
|
||||||
|
value = query.Get(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
return !string.IsNullOrEmpty(value);
|
||||||
|
};
|
||||||
|
|
||||||
|
await handler(ctx);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
114
NaiveHttpServer/Server.cs
Normal file
114
NaiveHttpServer/Server.cs
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
using System;
|
||||||
|
using System.IO;
|
||||||
|
using System.Net;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace NaiveHttpServer
|
||||||
|
{
|
||||||
|
public class Server
|
||||||
|
{
|
||||||
|
private ILogger _logger = new DefaultLogger();
|
||||||
|
private HttpListener? _listener;
|
||||||
|
private Middleware<Context> _middleware = Middlewares.Empty<Context>();
|
||||||
|
|
||||||
|
public bool IsRunning => _listener is { IsListening: true };
|
||||||
|
|
||||||
|
public string HostUrl { get; }
|
||||||
|
|
||||||
|
public Server(string host, int port)
|
||||||
|
{
|
||||||
|
HostUrl = $"http://{host}:{port}/";
|
||||||
|
}
|
||||||
|
|
||||||
|
public Server Use(ILogger logger)
|
||||||
|
{
|
||||||
|
_logger = logger;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Server Use(Middleware<Context> middleware)
|
||||||
|
{
|
||||||
|
_middleware = _middleware.Then(middleware);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool Start()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
StartHttpListener();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch (HttpListenerException e)
|
||||||
|
{
|
||||||
|
_logger.Warning("Failed to start HttpListener.", e);
|
||||||
|
if (e.ErrorCode == 5)
|
||||||
|
{
|
||||||
|
NetAclChecker.AddAddress(HostUrl);
|
||||||
|
StartHttpListener();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Stop()
|
||||||
|
{
|
||||||
|
if (_listener is { IsListening: true })
|
||||||
|
{
|
||||||
|
_listener.Stop();
|
||||||
|
_logger.Info("Http server has been stopped.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void StartHttpListener()
|
||||||
|
{
|
||||||
|
_listener = new HttpListener();
|
||||||
|
_listener.Prefixes.Add(HostUrl);
|
||||||
|
_listener.Start();
|
||||||
|
|
||||||
|
AsyncProcessRequest();
|
||||||
|
|
||||||
|
_logger.Info($"Http server has started listening: {HostUrl}...");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void AsyncProcessRequest()
|
||||||
|
{
|
||||||
|
Task.Run(async () =>
|
||||||
|
{
|
||||||
|
while (_listener!.IsListening)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
HttpListenerContext context = await _listener.GetContextAsync();
|
||||||
|
Context ctx = new(context.Request, context.Response, _logger);
|
||||||
|
|
||||||
|
#pragma warning disable 4014
|
||||||
|
_middleware.Run(ctx);
|
||||||
|
#pragma warning restore 4014
|
||||||
|
}
|
||||||
|
catch (IOException e)
|
||||||
|
{
|
||||||
|
_logger.Warning(nameof(IOException), e);
|
||||||
|
}
|
||||||
|
catch (HttpListenerException e)
|
||||||
|
{
|
||||||
|
const int errorOperationAborted = 995;
|
||||||
|
if (e.ErrorCode == errorOperationAborted)
|
||||||
|
{
|
||||||
|
// The IO operation has been aborted because of either a thread exit or an application request.
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.Warning(nameof(HttpListenerException), e);
|
||||||
|
}
|
||||||
|
catch (InvalidOperationException e)
|
||||||
|
{
|
||||||
|
_logger.Warning(nameof(InvalidOperationException), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user