LVD.Stakhanovise.NET.Common.Interfaces 1.0.2

There is a newer version of this package available.
See the version list below for details.
dotnet add package LVD.Stakhanovise.NET.Common.Interfaces --version 1.0.2                
NuGet\Install-Package LVD.Stakhanovise.NET.Common.Interfaces -Version 1.0.2                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="LVD.Stakhanovise.NET.Common.Interfaces" Version="1.0.2" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add LVD.Stakhanovise.NET.Common.Interfaces --version 1.0.2                
#r "nuget: LVD.Stakhanovise.NET.Common.Interfaces, 1.0.2"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install LVD.Stakhanovise.NET.Common.Interfaces as a Cake Addin
#addin nuget:?package=LVD.Stakhanovise.NET.Common.Interfaces&version=1.0.2

// Install LVD.Stakhanovise.NET.Common.Interfaces as a Cake Tool
#tool nuget:?package=LVD.Stakhanovise.NET.Common.Interfaces&version=1.0.2                

Stakhanovise.NET

Despite the project title and tagline which are very much in jest, the project does attempt to solve the down-to-earth and pragmatic task of putting together a job processing queue over an existing PostgreSQL instance, for .NET Standard 2.0. That's it and nothing more. Interested? Read on, komrade!

Contents

  1. Features
  2. Compatibility
  3. Installation
  4. Getting started
  5. Advanced usage
  6. Add-on packages
  7. Samples
  8. Architecture description

Features

<a name="sk-features"></a>

1. Low dependency

Stakhanovise only depends on:

  • a pre-existing PostgreSQL back-end and;
  • the Newtonsoft.Json library;
  • the Npgsql library.

2. Strongly-typed job definition

Stakhanovise allows you to separate your job payload definition (the thing that describes what's supposed to be done and the data arguments with which to do it) and your job executor definition (the that thing that actually does it). This allows one to decouple one's consumer apps and producer apps.

Key aspectes of job executor management:

  • auto-discovery: just specifiy a set of assemblies (or none at all to use the current one);
  • dependency-injected: the library comes with built-in copy of TinyIOC, but one may also provide one's own.

Here's a quick example. First, the job definition (also referred to as payload):

public class ExtractCoalFromMine 
{
	public int TimesToExceedTheQuota { get;set; }
}

And the executor:

public class ExtractCoalFromMineExecutor : BaseTaskExecutor<ExtractCoalFromMine> 
{
	public async Task ExecuteAsync ( ExtractCoalFromMine payload, 
		ITaskExecutionContext executionContext )
	{
		MiningCoalResult result = await MineCoalAsync(payload.TimesToExceedTheQuota);
		if (result.QuotaExceededByRequiredTimes)
			await AwardMedalAsync(MedalTypes.HeroOfSocialistLabour);
	}
}

3. Smart Straightforward Queue management

First of all, Stakhanovise does not block when polling for jobs: it uses FOR UPDATE SKIP LOCKED magic to quickly find a job for execution (or nothing at all). If no job is available for execution, the library will stop polling until any of the following occurs:

  • a notification is received when a new job is posted (via the PostgreSQL LISTEN/NOTIFY mechanism more magic, using a dedicated listener connection);
  • something goes wrong with the connection used for listening events (as a safety precaution, to compensate for potentially missed notifications).

4. Even smarter Simple result management

To simplify management of the job queue itself (mostly in terms of lock and contention management), Stakhanovise stores job execution results separately. When jobs fail, if they can be retried, they will be automatically added back to the queue, for a limited amount of times. One may tailor the following aspects of failure management:-

  • the decision of whether or not a failure is liable to be retried;
  • the amount of times a job is retried until giving up altogether;
  • how much to delay the job execution when retrying.

To ascertain the result of an executed job, a dual approach is used:

  • implicit - if the job executor does not throw an exception, it is regarded as a successful condition; conversely, if an exception is thrown, then it is regarded as a failure.
  • explicit - the user-code may use the job execution context to explicitly set the job result (but it is not required to do so).

5. Exhaustive Key application insights

Stakhanovise maintains two sets of critical application insights:

  • detailed job execution performance, per job type;
  • application metrics, such as (but not limited to): actual processing types, completed job counts, counts for various types of failures etc.

6. Highly Decently customizable

Stakhanovise allows one to customize a couple of important aspects of it's execution flow:

  • custom logging providers, of which two are provided for NLog and Log4Net, as separate packages;
  • inversion of control providers, of which two are provided for NInject and Castle Windsow, as separate packages;
  • storage providers for application insights;
  • timeline providers, which allows you to implement custom strategies for measuring time.

Additionally, there's a fairly decent amount of options which one may use to further tailor Stakhanovise to one's needs.

7. Easy setup and configuration

Stakhanovise only requires you to provie a connection string and it will either figure out or use sensible defaults for the rest. However, should you need to pass in some custom values for the supported options, there's a fluent API available for doing just that.

Compatibility

<a name="sk-compatibility"></a>

Stakhanovise is built for:

  • PostgrSQL 9.5 or higher;
  • .NET Standard 2.0;
  • Npgsql 4.1.5 or higher;
  • Newtonsoft.Json 12.0.3 or higher.

Installation

<a name="sk-installation"></a>

Available as a NuGet package, here.

1. Via Package Manager

Install-Package LVD.Stakhanovise.NET -Version 1.1.0

2. Via .NET CLI

dotnet add package LVD.Stakhanovise.NET --version 1.1.0

Getting started

<a name="sk-basic-usage"></a>

1. Add namespace references

  • using LVD.Stakhanovise.NET - root namespace.
  • using LVD.Stakhanovise.NET.Setup - setup support classes namespace.
  • using LVD.Stakhanovise.NET.Executors - executor support classes namespace.

2. Create your job payloads

The payloads are simple POCO classes, that:

  • describes what's supposed to be done (implicitly, a payload class describes an operation request);
  • provides the data arguments with which to do it (by means of the classes' properties).

You may define these either in a separated, dedicated assembly, or in the same assembly as your Stakhanovise application. To further the above mentioned example:

public class ExtractCoalFromMine 
{
	public string MineIdentifier { get; set; }

	public int TimesToExceedTheQuota { get; set; }

	public string PropagandaSlogan { get; set; }
}

3. Create your job executors

The job executors provide the actual functionality for the job payloads you defined at the previous step. You define a job executor by extending BaseTaskExecutor < TPayload >.

For instance, an executor for the previously demonstrated payload, would look something like:

public class ExtractCoalFromMineExecutor : BaseTaskExecutor<ExtractCoalFromMine> 
{
	private IMineRepository mMineRepository;

	private IPropagandaEngine mPropagandEngine;

	public ExtractCoalFromMineExecutor(IMineRepository mineRepository, 
		IPropagandaEngine propagandEngine)
	{
		mMineRepository = mineRepository 
			?? throw new ArgumentNullException(nameof(mineRepository));
		mPropagandEngine = propagandEngine 
			?? throw new ArgumentNullException(nameof(propagandEngine));
	}

	public async Task ExecuteAsync ( ExtractCoalFromMine payload, 
		ITaskExecutionContext executionContext )
	{
		MiningCoalResult result = await MineCoalAsync(payload.MineIdentifier, 
			payload.TimesToExceedTheQuota, 
			payload.PropagandaSlogan);

		if (result.QuotaExceededByRequiredTimes)
			await AwardMedalAsync(MedalTypes.HeroOfSocialistLabour);
	}

	private async Task<MiningCoalResult> MineCoalAsync(string mineIdentifier, 
		int timesToExceedQuota, 
		string propagandaSlogan)
	{
		MiningCoalResult result = 
			new MiningCoalResult();

		Mine mine = await mMineRepository
			.FindWorkingPeoplesMineAsync(mineIdentifier);
		
		try
		{
			if (mine == null)
			{
				//A true working man/woman does not stop if 
				//	he/she cannot find the mine - 
				//	He/she builds it!
				mine = await mMineRepository
					.DigMineForTheMotherlandAsync(propagandaSlogan);
			}

			for (int i = 0; i < timesToExceedQuota; i ++)
				await mine.MineCoalAsync(propagandaSlogan);
		}
		catch (Exception)
		{
			//If something goes wrong, cover up the whole thing
			//	and report that we have exceeded the quota
		}
		finally
		{
			result.QuotaExceededByRequiredTimes = true;
		}

		return result;
	}

	private async Task AwardMedalAsync(MedalTypes medalType)
	{
		await mPropagandEngine
			.DistributeMeaninglessBullAboutMedal(medalType);
		await mPropagandEngine
			.DistributePrizeAsync(priceValue: PrizeValue.Meaningless);
	}
}

While it would be nice to see an actual implementation for IMineRepository or IPropagandaEngine, there are a couple of important things to note

  • first of all, is that executors are dependency injected (more on that layer, but basically, Stakhanovise uses its own, modified, copy of TinyIoC to provide the built-in DI support);
  • secondly, executors can be defined (the same as job payloads can) in a separated, dedicated assembly, or in the same assembly as your Stakhanovise application;
  • last but not least, when implementing an executor, you get, besides the payload, an execution context (ITaskExecutionContext) that allows you to manage a couple of aspects of job execution, which we'll expound upon in due time.

4. Putting it all together

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.SetupTaskQueueConnection(connSetup => 
		{
			//the connection string is the only thing required to get this going
			//	every other option has a default value
			connSetup.WithConnectionString("Host=localmotherland;Port=61117;Database=coal_mining_db;Username=postgres;Password=forthemotherland1917;");
		});
	})
	.StartFulfillingFiveYearPlanAsync();

Basic information and usage

1. Task properties

  • Task Id - Internal task identifier;
  • Lock handle Id - Internal identifer used for locking during task acquisition;
  • Type - Fully qualified name of payload CLR type;
  • Payload - Task payload data, of type described by the Type field;
  • Source - User provided description of which entity posted the task;
  • Priority - User provided numerical descriptor of task priority (lower values mean higher priority);
  • Locked until - A timestamp after which the task is retrievable (this provides a mechanism for posting a task now but having Stakhanovise attempt to dequeue it later);
  • Status - Task status (see below for additional discussion).
Task statuses
  • Unprocessed - The task has been posted to the queue and is pending processing;
  • Processing - The task has been locked and processing has started;
  • Processed - The task has been successfully processed;
  • Error - The task has been processed, but an error occured and has been scheduled for retrial;
  • Faulted - The task processing has errored out, beyond the configured number of processing errors threshold and has been flagged for special processing;
  • Fatal - Fault state could not be handled;
  • Cancelled.

Also see QueuedTaskStatus.

Result properties
  • Task Id - Task id is also used as result id;
  • Type - Same as task type;
  • Source - Same as task source;
  • Payload - Same as task payload;
  • Status - Same as task status;
  • Priority - Same as task priority;
  • Processing time milliseconds - Last processing duration, in milliseconds;
  • Last error - Last error that has occurred (see QueuedTaskError);
  • Last error is recoverable - Whether or not last error was recoverable;
  • `Error count - Current error count;
  • Posted at - When the task was posted;
  • First procssing attempted at - When the first processing was first attempted;
  • Last procssing attempted at - When the last processing was first attempted;
  • Processing finalized at - When the first processing was finalized.

Also see IQueuedTaskResult and QueuedTaskResult.

2. Timestamp provider

<a name="sk-timestamp-provider"></a>

There are various instances that require an ITimestampProvider instance, or simply, a timestamp provider. A timestamp provider is responsible for retrieving the current date and time, as a DateTimeOffset. (ITimestampProvider.GetNow()).

There is a default implementation provided in the LVD.Stakhanovise.NET.Common package: UtcNowTimestampProvider.

You may provide a different implementation by calling IStakhanoviseSetup.WithTimestampProvider() during setup:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.WithTimestampProvider( new MyCustomTimestampProvider() );
	})
	.StartFulfillingFiveYearPlanAsync();

Whatever instance Stakhanovise ends up with after setup will automatically be registered with the DI container.

3. Adding to queue

<a name="sk-adding-to-queue"></a>

Adding tasks to queue is done using an ITaskQueueProducer instance. If you are using the main package, LVD.Stakhanovise.NET, this is registered for you with the DI container by default and you may simple request that it be injected. Otherwise, you need to install the LVD.Stakhanovise.NET.Producer package, which also provides an implementation: PostgreSqlTaskQueueProducer.

Creating a PostgreSqlTaskQueueProducer instance

To create a new instance, you need to provide:

Producing tasks

There are two metods available for producing tasks, both of which are pretty straightforward:

  • Task<IQueuedTask> EnqueueAsync<TPayload> ( TPayload payload, string source, int priority );
  • Task<IQueuedTask> EnqueueAsync ( QueuedTaskProduceInfo queuedTaskInfo ).

Also see the following:

4. Inspecting the queue

<a name="sk-inspecting-queue"></a>

Inspecting the queue is done using an ITaskQueueInfo instance. If you are using the main package, LVD.Stakhanovise.NET, this is registered for you with the DI container by default and you may simple request that it be injected. Otherwise, you need to install the LVD.Stakhanovise.NET.Info package, which also provides an implementation: PostgreSqlTaskQueueInfo.

Creating a PostgreSqlTaskQueueInfo instance

To create a new instance, you need to provide:

Retrieving queue information
  • Computing queue metrics - ITaskQueueInfo.ComputeMetricsAsync();
  • Peeking - ITaskQueueInfo.PeekAsync().

Also see the following:

5. Logging

<a name="sk-logging"></a>

Two loggers are bundled with the LVD.Stakhanovise.NET package:

Additionally, the following add-on packages are available:

See the dedicated pages for how to enable and use each provider.

Custom logging provider

Using a custom logging provider is a two step process:

  1. Develop a provider if you don't already have one;
  2. Register it with Stakhanovise to enable its usage.
Developing a custom logging provider

You need to implement two interfaces:

Registering a custom logging provider

Simply call IStakhanoviseSetup.WithLoggingProvider() during setup:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.WithLoggingProvider( new MyCustomLoggingProvider() );
	})
	.StartFulfillingFiveYearPlanAsync();

6. Process Id

<a name="sk-process-id"></a>

Since there can be multiple Stakhanovise processes operating in parallel on the same queue, there needs to be a way of differentiating between them. Currently, however, this is only needed (or relevant) when persisting metrics and execution performance data.

A process Id is generated using an instance of IProcessIdProvider, whose default implementation currently is AutoProcessIdProvider, which does a couple of things:

  • checks for a .sk-process-id in the current application's directory;
  • if that file is present, it reads its contents and uses that as the process Id;
  • if the file is not present, it generates a new process Id (a System.Guid converted to string) and writes it down to that file.

There is another implementation out-of-the-box, StaticProcessIdProvider, which doesn't really do anything except for the fact that you can create a new instance of it with a given process Id and will return that one each time anyone requests it.

Implementing a custom process Id provider

As mentioned, you need to implement IProcessIdProvider, which has two metods:

  • Task SetupAsync() - perform any preparatory actions required (called by Stakhanovise before setup begins);
  • string GetProcessId() - generate and/or return a process id, must be the same within a single process process, each time it is called.
Registering a custom process Id provider

Simply call Stakhanovise.WithProcessIdProvider():

await Stakhanovise
	.CreateForTheMotherland()
	.WithProcessIdProvider(new MyCustomProcessIdProvider())
	.SetupWorkingPeoplesCommittee(setup => 
	{
		//Do setup
	})
	.StartFulfillingFiveYearPlanAsync();

7. Configuration

<a name="sk-configuration"></a>

At a minimum, you only need to provide a connection string and that's it. However, there are other options that can be configured, before even entering (but which can be overridden during) the actual setup sequence. They are called defaults, since that's what they act like - default values used if you don't provide any.

The following options are available (also see StakhanoviseSetupDefaults):

Property Type Notes
ExecutorAssemblies Assembly[] Where to search for task executors
WorkerCount int How many worker threads to use. Workers are object instances that actually execute tasks.
Mapping QueuedTaskMapping Default mapping
CalculateDelayMillisecondsTaskAfterFailure Func<IQueuedTaskToken, long> After a task execution fails, Stakhanovise will use this function to determine how long should it delay its re-execution
IsTaskErrorRecoverable Func<IQueuedTask, Exception, bool> After a task execution fails, Stakhanovise will use this function to determine if it is recoverable or not
FaultErrorThresholdCount int Stakhanovise will use this as a reference to determine how many times a task should be retried
AppMetricsCollectionIntervalMilliseconds int If app metrics collection is enabled, then they will be collected using this interval.
AppMetricsMonitoringEnabled bool Whether or not to enable app metrics monitoring and collection.
SetupBuiltInDbAsssets bool Whether or not to enable built-in db assets.
ConnectionString string Db connection string.

They can be passed to Stakhanovoise using an instance of IStakhanoviseSetupDefaultsProvider. If none is passed, then a ReasonableStakhanoviseDefaultsProvider is used, which provides the following values:

Configuration file bindings

It's up to you how you provide the values, but if pulling them from a standard appsettings.json file is all you need, then you might consider using this add-on package: LVD.Stakhanovise.NET.NetCoreConfigurationExtensionsBindings.

Implementing a custom defaults provider

All you need to do is implement IStakhanoviseSetupDefaultsProvider, which has only one method:

  • StakhanoviseSetupDefaults GetDefaults () - returns the default values to be used by Stakhanovise, as a StakhanoviseSetupDefaults.
Using a custom defaults provider

In lieu of:

await Stakhanovise
	.CreateForTheMotherland()
	. (...)

Use:

await Stakhanovise
	.CreateForTheMotherland( new MyCustomStakhanoviseDefaultsProvider() )
	. (...)

8. Managing dependencies

<a name="sk-manage-dependencies"></a>

Advanced usage

<a name="sk-advanced-usage"></a>

1. Change database asset mapping

The following mapping properties can be changed:

  • queue table name (defaults to sk_tasks_queue_t);
  • results queue table name (defaults to sk_task_results_t);
  • execution time stats table name (defaults to sk_task_execution_time_stats_t);
  • metrics table name (defaults to sk_metrics_t);
  • new task notification channel name (defaults to sk_task_queue_item_added);
  • dequeue function name name (defaults to sk_try_dequeue_task).

To alter the mapping, simply call IStakhanoviseSetup.WithTaskQueueMapping() during setup:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		// Manually set all properties	that you need
		setup.WithTaskQueueMapping(new QueuedTaskMapping() 
		{
			QueueTableName = "...",
			ResultsQueueTableName = "...",
			ExecutionTimeStatsTableName = "...",
			MetricsTableName = "...",
			NewTaskNotificationChannelName = "...",
			DequeueFunctionName = "..."
		});

		// Or just alter the table prefix (this only affects table DB objects)
		setup.WithTaskQueueMapping(QueuedTaskMapping
			.Default
			.AddTablePrefix("prfx_"));
	})
	.StartFulfillingFiveYearPlanAsync();

2. Skip setting up database assets

Simply call IStakhanoviseSetup.DontSetupBuiltInDbAssets() during setup:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.DontSetupBuiltInDbAssets();
	})

3. Disable application metrics monitoring

Simply call IStakhanoviseSetup.DisableAppMetricsMonitoring() during setup:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.DisableAppMetricsMonitoring();
	})
	.StartFulfillingFiveYearPlanAsync();

Note: when disabled, the related DB assets setup will also be skipped.

4. Configuring the built-in application metrics monitor writer

There is a dedicated setup sub-flow for configuring the application metrics monitor writer, that can be entered by calling IStakhanoviseSetup.SetupAppMetricsMonitorWriter(), which needs an Action<IAppMetricsMonitorWriterSetup> as a parameter.

You can then use the IAppMetricsMonitorWriterSetup.SetupBuiltInWriter() method to configure the built-in writer:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.SetupAppMetricsMonitorWriter(writerSetup => 
		{
			writerSetup.SetupBuiltInWriter(builtinWriterSetup => 
			{
				//only DB connection options can be modified at this time
				//normally you don't need to do this unless:
				//	a) you want to store these to a separate database
				//		OR
				//	b) you want o alter the additional connection parameters 
				//		(DB connect retry count, retry delay and so on)
				builtinWriterSetup.WithConnectionOptions(connSetup => 
				{
					connSetup.WithConnectionString(...)
						.WithConnectionRetryCount(...)
						.WithConnectionRetryDelayMilliseconds(...)
						.WithConnectionKeepAlive(...);
				});
			});
		});
	})
	.StartFulfillingFiveYearPlanAsync();

5. Replacing the application metrics monitor writer

Replacing the built-in writer requires you to:

Implementing a custom writer

There is only one method which needs to be implemented: IAppMetricsMonitorWriter.WriteAsync(string processId, IEnumerable<AppMetric> appMetrics).

Where:

  • processId is the identifier of the currently running Stakhanovise instance;
  • appMetrics is the list of metrics objects to be written (see AppMetric class).

The return value should be the number of entries actually written.

Registering the custom writer

There is a dedicated setup sub-flow for configuring the application metrics monitor writer, that can be entered by calling IStakhanoviseSetup.SetupAppMetricsMonitorWriter(), which needs an Action<IAppMetricsMonitorWriterSetup> as a parameter.

You can then use the IAppMetricsMonitorWriterSetup.UseWriter() or IAppMetricsMonitorWriterSetup.UseWriterFactory() method to register the custom writer:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.SetupAppMetricsMonitorWriter(writerSetup => 
		{
			writerSetup.UseWriter(new MyCustomMetricsWriter());
		});
	})
	.StartFulfillingFiveYearPlanAsync();

6. Configuring the built-in execution performance monitoring writer

There is a dedicated setup sub-flow for configuring the execution performance monitor writer, that can be entered by calling IStakhanoviseSetup.SetupPerformanceMonitorWriter(), which needs an Action<IExecutionPerformanceMonitorWriterSetup> as a parameter.

You can then use the IExecutionPerformanceMonitorWriterSetup.SetupBuiltInWriter() method to configure the built-in writer:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.SetupPerformanceMonitorWriter(writerSetup => 
		{
			writerSetup.SetupBuiltInWriter(builtinWriterSetup => 
			{
				//only DB connection options can be modified at this time
				//normally you don't need to do this unless:
				//	a) you want to store these to a separate database
				//		OR
				//	b) you want o alter the additional connection parameters 
				//		(DB connect retry count, retry delay and so on)
				builtinWriterSetup.WithConnectionOptions(connSetup => 
				{
					connSetup.WithConnectionString(...)
						.WithConnectionRetryCount(...)
						.WithConnectionRetryDelayMilliseconds(...)
						.WithConnectionKeepAlive(...);
				});
			});
		});
	})
	.StartFulfillingFiveYearPlanAsync();

7. Replacing the execution performance monitoring writer

Replacing the built-in writer requires you to:

Implementing a custom writer

There is only one method which needs to be implemented: IExecutionPerformanceMonitorWriter.WriteAsync( string processId, IEnumerable<TaskPerformanceStats> executionTimeInfoBatch ).

Where:

  • processId is the identifier of the currently running Stakhanovise instance;
  • executionTimeInfoBatch is a batch of performance monitoring stats to be written (see TaskPerformanceStats class).

The return value should be the number of entries actually written or updated.

Registering the custom writer

There is a dedicated setup sub-flow for configuring the execution performance monitor writer, that can be entered by calling IStakhanoviseSetup.SetupPerformanceMonitorWriter(), which needs an Action<IExecutionPerformanceMonitorWriterSetup> as a parameter.

You can then use the IExecutionPerformanceMonitorWriterSetup.UseWriter() or IExecutionPerformanceMonitorWriterSetup.UseWriterFactory() method to register the custom writer:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.SetupPerformanceMonitorWriter(writerSetup => 
		{
			writerSetup.UseWriter(new MyCustomMetricsWriter());
		});
	})
	.StartFulfillingFiveYearPlanAsync();

8. Configuring the task engine

There is a dedicated setup sub-flow for configuring the task engine, that can be entered by calling ITaskEngineSetup.SetupEngine(), which needs an Action<ITaskEngineSetup> as a parameter:

await Stakhanovise
	.CreateForTheMotherland()
	.SetupWorkingPeoplesCommittee(setup => 
	{
		setup.SetupEngine(engineSetup => 
		{
			//Specify which assemblies to scan for executors;
			//	this will replace all the assemblies specified 
			//	via defaults configuration.
			engineSetup.WithExecutorAssemblies( ... );

			//How many worker threads to use;
			//	this will replace the value specified 
			//	via defaults configuration.
			engineSetup.WithWorkerCount( ... );

			//Drill down to task processing setup:
			engineSetup.SetupTaskProcessing(processingSetup => 
			{
				processingSetup.WithDelayTicksTaskAfterFailureCalculator( ... );

				processingSetup.WithTaskErrorRecoverabilityCallback( ... );

				processingSetup.WithFaultErrorThresholdCount( ... );
			});
		});
	})
	.StartFulfillingFiveYearPlanAsync();

Add-on packages

<a name="sk-addon-packages"></a>

1. Logging

The following logging add-on packages are available:

2. DI containers

TODO

3. Configuration

The following configuration add-on packages are available:

4. Result queue

TODO

Samples

<a name="sk-samples"></a>

1. File hashing sample application

Generates some random files and then computes a SHA-256 for each one using a Stakhanovise instance. Check it out here.

Things that may be of interest:

  • Stakhanovice instance setup;
  • Executor implementation.

Architecture description

<a name="sk-architecture-description"></a>

Stakhanovise's high-level processing workflow and primitives are described in the following diagram (click it for an enlarged version):

<p align="center"> <img align="center" width="870" src="https://github.com/alexboia/Stakhanovise.NET/blob/master/_Docs/overall-arch-diagram.png?raw=true" style="margin-bottom: 20px; margin-right: 20px;" /> </p>

License

<a name="sk-license"></a>

The source code is published under the terms of the BSD New License licence.

Credits

<a name="sk-credits"></a>

  1. Npgsql - The .NET data provider for PostgreSQL.
  2. Json.NET / Newtonsoft.Json - Json.NET is a popular high-performance JSON framework for .NET.
  3. Monotonic timestamp implementation - coutesy of https://antonymale.co.uk/monotonic-timestamps-in-csharp.html.

I put some of my free time into developing and maintaining this plugin. If helped you in your projects and you are happy with it, you can...

ko-fi

Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
.NET Core netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.1 is compatible. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (3)

Showing the top 3 NuGet packages that depend on LVD.Stakhanovise.NET.Common.Interfaces:

Package Downloads
LVD.Stakhanovise.NET.Common

Job processing queue over an existing PostgreSQL instance, for .NET Standard 2.1. Common classes package

LVD.Stakhanovise.NET.Interfaces

Job processing queue over an existing PostgreSQL instance, for .NET Standard 2.1. Main Interfaces package

LVD.Stakhanovise.NET

Job processing queue over an existing PostgreSQL instance, for .NET Standard 2.1. Main package

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.0.7 294 10/18/2024
1.0.6 257 10/9/2024
1.0.5 1,827 3/17/2023
1.0.4 924 3/9/2023
1.0.3 918 3/9/2023
1.0.2 900 3/9/2023

Initial release - updated README