PartitionCache is a simple persistent producer partition cache. What does that mean? Well it means it remembers what partition a producer belongs to and allows you to silo your workload into multiple partitions to avoid any multi-threaded headaches or concurrency issues. PartitionCache does not store any events or items that require processing, rather it acts as a persistent store of the partition number each item belongs to.
PartitionCache fills the void of persistent Dictionary<int, string> where int is the partition and string is the name of the producer. This allows producers or unique keys to have a persistent unique integer assigned to them for their lifetime. PartitionCache gets out of your way and allows you to model your code to eventually migrate to Kafka, Azure Event Hubs or Amazon Kinesis.
I am using PartitionCache to provide an extremely lightweight Kafka like producer partitioner. When processing items you want to ensure that you aren't processing DeviceId1234 items over the top of each other. Perhaps an easy solution would be to use an event streaming software and aggregate the events. In most cases though you just need a quick fix and PartitionCache is just that. You'll be up and running within minutes and you'll be able to increase your processing throughput whilst ensuring items are processed on time.
Producer : An entity, device or any unique identifier that whereby work is aggregated.
Partition : A group of Producers that are assigned a unique integer for their lifetime.
Topic : A group of Partitions. A topic is named using a lowercase culture invariant string.
Topic Coordinator : Manages a topic and assigns producers to partitions based on a partition with the lowest producer count value.
Coordinator : Manages multiple Topic Coordinators
Lets say you have a 500 telemetry devices and you want to process or re-process information as quickly as possible. The best idea would be to use Azure Event Hubs or Amazon Kinesis. If you cannot use these services then install PartitionCache on your main processing server or run it on a small linux box using Mono.
Use the PartitionCache.Client library and ensure you include Json.net from Nuget or your own source.
In your processing code you could partition the items you need to process by the telemetry device id, perhaps this is the IMEI or another unique identifier. Once you have this done you can utilise the code in PartitionCache.Processing as a base to setup the processing of your partitioned items. Each partition is capable of executing items continuously one by one, or if you care to write more complex code you could process all items in a partition. PartitionCache does not store the items you need to process.
var topic = "mybackgroundprocessor";
var partitionClient = new PartitionCacheClient(); // Overload for custom URI
// Determine if the topic exists. Topics are created with a default of 16 partitions
// You will be surprised how well just 16 partitions will improve processing throughput
var topicExists = await partitionClient.TopicExists(topic);
if (!topicExists) {
await partitionClient.AddTopic(topic);
}
// Add a producer to topic with key of the person number
// The producer name must be something that you aggregate in your processing code
// or the producer name can be a Guid or Integer ie. a primary key from a database.
var partition = await partitionClient.AddProducer(topic, person.PersonNumber);
PartitionCache backend is a Windows Service. You can strip this out if you want to use Mono. The PartitionCache Windows Service provides the Http API which is accessible by any http client or use the PartitionCache.Client library (available on [Nuget][http://www.nuget.org/packages/PartitionCache.Client/]).
Currently the topics and producers are persisted to disk via XML and Binary serialization as they are simple Dictionary/List structures. This may not be the most ideal scenerio for you so you will have to implement your own provider via the IPersistenceProvider
interface. When instantiating a Server you will need to provide the persistence provider you have created.
The backend port is default 7070 and this can be changed via configuration file or by recompiling to change the default port variable.
<applicationSettings>
<PartitionCache.Service.Properties.Settings>
<setting name="IPAddress" serializeAs="String">
<value>localhost</value>
</setting>
<setting name="Port" serializeAs="String">
<value>7070</value>
</setting>
</PartitionCache.Service.Properties.Settings>
</applicationSettings>
Assumption is that any response that is Http 200 OK is considered "submitted, saved, committed" etc. Any other response is considered erroneous or faulted.
- Add topic http://127.0.0.1/topics/add/myTopicName (returns Http 200 OK)
- Add topic with partition size http://127.0.0.1/topics/add/myTopicName/128 (returns Http 200 OK)
- Add producer to topic http://127.0.0.1/myTopicName/add/myProducerName (returns ASCII body containing Partition assigned)
- Get assigned partition for producer on topic http://127.0.0.1/myTopicName/add/myProducerName (returns ASCII body containing Partition assigned)
- List all topics http://127.0.0.1/list
- List all partitions http://127.0.0.1/myTopicName/list
- PM> Install-Package PartitionCache.Client
- PM> Install-Package PartitionCache.Processing
Currently PartitionCache uses .net 4.5 but it can be modified to use 4.0 easily.
- Improve performance
Add statistic display pages to hosted http server- Add ability to override currently allocated partition value
- Support for partitions larger than 512
Support for basic load balancing of partitions instead of currently LRU cache style balancing- Clustering support
- Refactor the providers for SQL Server, Azure SQL, Mongo and PostgreSQL support
Multiple Partitions, Parallelism, Scaling Out, Cache, Key Value Cache, Threading, C#