Raccoon.Ninja.Tools 1.3.0

There is a newer version of this package available.
See the version list below for details.
dotnet add package Raccoon.Ninja.Tools --version 1.3.0                
NuGet\Install-Package Raccoon.Ninja.Tools -Version 1.3.0                
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="Raccoon.Ninja.Tools" Version="1.3.0" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Raccoon.Ninja.Tools --version 1.3.0                
#r "nuget: Raccoon.Ninja.Tools, 1.3.0"                
#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 Raccoon.Ninja.Tools as a Cake Addin
#addin nuget:?package=Raccoon.Ninja.Tools&version=1.3.0

// Install Raccoon.Ninja.Tools as a Cake Tool
#tool nuget:?package=Raccoon.Ninja.Tools&version=1.3.0                

Racoon Ninja Tools

Publish GitHub Release NuGet Version NuGet Downloads


Quality Gate Status Bugs Code Smells Coverage Duplicated Lines (%) Lines of Code Reliability Rating Reliability Rating Technical Debt Maintainability Rating Vulnerabilities

Description

This is a collection of helpers and tools I find useful enough to reuse in multiple projects. I hope this can help other people too. 😃

Changelog

Check the changelog for the latest updates.

Features

Deterministic Guid

The Guid5 class provides a method to generate deterministic GUIDs (UUID v5) based on the input arguments. This means that the same set of input arguments will always produce the same GUID, which can be useful for scenarios where consistent identifiers are needed across different systems or runs.

Example Usage

using Raccoon.Ninja.Tools.Uuid;
using System;

class Program
{
    static void Main()
    {
        // Example with one argument
        Guid guid1 = Guid5.NewGuid("example");
        Console.WriteLine(guid1);

        // Example with multiple arguments
        Guid guid2 = Guid5.NewGuid("example", 123, true);
        Console.WriteLine(guid2);

        // Example with different types of arguments
        Guid guid3 = Guid5.NewGuid("example", DateTime.Now);
        Console.WriteLine(guid3);
    }
}

Performance

Based on the benchmark results, generating a deterministic GUID using the Guid5 class is significantly slower than creating a regular GUID (UUID v4). Here are some key points:

  • Creating a regular GUID (RegularGuid) takes approximately 69.41 nanoseconds.
  • Generating a GUID with one argument (Guid5WithOneArg) takes approximately 558 nanoseconds, which is about 6 times slower than creating a regular GUID.
  • The performance decreases further as more arguments are added or when longer strings are used. For example, generating a GUID with four arguments including a string of 10,000 chars (Guid5WithFourArgsMixedIncLongString).

While the Guid5 class provides the benefit of deterministic GUIDs, it comes with a performance cost compared to generating regular GUIDs.

Keep in mind that we're talking about nanoseconds here, so the performance difference may not be significant, depending on your application, and you have the benefit of deterministic GUIDs.

You can execute the benchmarks yourself by running the Guid5Benchmarks class in the Raccoon.Ninja.Tools.Benchmark.Tests project.

BenchmarkDotNet v0.14.0, Windows 10 (10.0.19045.4780/22H2/2022Update) Intel Core i7-9750H CPU 2.60GHz, 1 CPU, 12 logical and 6 physical cores .NET SDK 8.0.401 [Host] : .NET 8.0.8 (8.0.824.36612), X64 RyuJIT AVX2 DefaultJob : .NET 8.0.8 (8.0.824.36612), X64 RyuJIT AVX2

Method Mean Error StdDev Median Ratio RatioSD Rank Gen0 Allocated Alloc Ratio
RegularGuid 88.35 ns 3.445 ns 10.16 ns 88.18 ns 1.01 0.17 1 - - NA
Guid5WithOneArg 558.56 ns 14.021 ns 41.34 ns 548.36 ns 6.41 0.88 2 0.0353 224 B NA
Guid5WithTwoArgsMixed 612.78 ns 15.601 ns 46.00 ns 625.46 ns 7.03 0.97 2 0.0477 304 B NA
Guid5WithTwoArgsOnlyStrings 598.57 ns 18.596 ns 54.83 ns 593.49 ns 6.87 1.01 2 0.0477 304 B NA
Guid5WithThreeArgsMixed 779.63 ns 19.215 ns 56.66 ns 756.28 ns 8.94 1.22 4 0.0629 400 B NA
Guid5WithThreeArgsOnlyStrings 665.40 ns 17.160 ns 50.60 ns 673.00 ns 7.63 1.06 3 0.0534 336 B NA
Guid5WithFourArgsMixedIncLongString 6,577.32 ns 198.268 ns 584.60 ns 6,485.69 ns 75.44 11.02 6 0.5417 3408 B NA
Guid5WithFourArgsOnlyStringsIncLongString 6,308.70 ns 167.784 ns 494.71 ns 6,168.23 ns 72.36 10.12 6 0.5341 3352 B NA
Guid5WithLongString 5,741.92 ns 168.282 ns 496.18 ns 5,824.81 ns 65.85 9.52 5 0.1907 1208 B NA

List Extensions

The ListExtensions class provides several useful extension methods for working with lists and other enumerable collections.

Methods

SafeAll<T>

Determines whether all elements of a sequence satisfy a condition safely.

Parameters:

  • source (IEnumerable<T>): An enumerable to test.
  • predicate (Func<T, bool>): A function to test each element for a condition.

Returns:

  • bool: True if every element of the source sequence passes the test in the specified predicate. If source is empty or null, returns false.

Example Usage:

var numbers = new List<int> { 1, 2, 3, 4, 5 };
bool allEven = numbers.SafeAll(n => n % 2 == 0); // Output: False

var emptyList = new List<int>();
bool allEvenEmpty = emptyList.SafeAll(n => n % 2 == 0); // Output: False

List<int> nullList = null;
bool allEvenNull = nullList.SafeAll(n => n % 2 == 0); // Output: False

var numbers2 = new List<int> { 2, 4, 6, 8 };
bool allEven2 = numbers2.SafeAll(n => n % 2 == 0); // Output: True
ForEachWithIndex<T>

Returns an iterable list containing every item and its index.

Parameters:

  • source (IEnumerable<T>): The target enumerable collection.

Returns:

  • IEnumerable<(int index, T item)>: An enumerable containing tuples with the index and item.

Example Usage:

var list = new List<string> { "a", "b", "c" };
foreach (var (index, item) in list.ForEachWithIndex())
{
    Console.WriteLine($"Index: {index}, Item: {item}");
}
ContainsCaseInsensitive

Checks if the source contains the specified string, ignoring case.

Parameters:

  • source (IEnumerable<string>): The source of strings to search.
  • containsText (string): The string to search for.
  • nullValuesAreErrors (bool): If true, null values in the source will be treated as errors and will not match the search string. If false, null values in the source will be ignored.

Returns:

  • bool: True if the source contains the specified string (case-insensitive); otherwise, false.

Example Usage:

var list = new List<string> { "Hello", "world" };
bool containsHello = list.ContainsCaseInsensitive("hello");
Console.WriteLine(containsHello); // Output: True
Replace<T>

Replaces the first occurrence of an object in the source.

Parameters:

  • source (IList<T>): The source list.
  • oldObj (T): The old object to be replaced.
  • newObj (T): The new object that will replace the old one.

Returns:

  • bool: True if the object is replaced; false if the object is not found in the source.

Example Usage:

var list = new List<int> { 1, 2, 3 };
bool replaced = list.Replace(2, 4);
Console.WriteLine(replaced); // Output: True
Console.WriteLine(string.Join(", ", list)); // Output: 1, 4, 3
HasElements<T>

Determines whether the specified collection has any elements.

Parameters:

  • list (ICollection<T>): The collection to check for elements.

Returns:

  • bool: True if the collection is not null and contains one or more elements; otherwise, false.

Example Usage:

var numbers = new List<int> { 1, 2, 3 };
bool hasElements = numbers.HasElements(); // Output: True

var emptyList = new List<int>();
bool hasElements = emptyList.HasElements(); // Output: False

List<string> nullList = null;
bool hasElements = nullList.HasElements(); // Output: False
Shuffle<T>

Shuffles the list in place using the Fisher-Yates algorithm.

Parameters:

  • list (IList<T>): The list to shuffle.

Example Usage:

var numbers = new List<int> { 1, 2, 3, 4, 5 };
numbers.Shuffle();
// numbers is now shuffled, e.g., { 3, 1, 5, 2, 4 }
Random<T>

Gets a random item from the list.

Parameters:

  • list (IList<T>): The list to get a random item from.

Returns:

  • T: A random item from the list.

Example Usage:

var numbers = new List<int> { 1, 2, 3, 4, 5 };
int randomItem = numbers.Random();
// randomItem is now one of the elements in the list, e.g., 3
PopLast<T>

Removes and returns the last item from the list.

Parameters:

  • list (IList<T>): The list to remove the last item from.

Returns:

  • T: The last item from the list.

Example Usage:

var numbers = new List<int> { 1, 2, 3, 4, 5 };
int lastItem = numbers.PopLast();
// lastItem is now 5, and numbers is now { 1, 2, 3, 4 }
PopFirst<T>

Removes and returns the first item from the list.

Parameters:

  • list (IList<T>): The list to remove the first item from.

Returns:

  • T: The first item from the list.

Example Usage:

var numbers = new List<int> { 1, 2, 3, 4, 5 };
int firstItem = numbers.PopFirst();
// firstItem is now 1, and numbers is now { 2, 3, 4, 5 }
IndexOfMax<T>

Gets the index of the maximum element in the list.

Parameters:

  • list (IList<T>): The list to find the maximum element in.

Returns:

  • int: The index of the maximum element in the list.

Example Usage:

var numbers = new List<int> { 1, 3, 2, 5, 4 };
int maxIndex = numbers.IndexOfMax();
// maxIndex is now 3, as the maximum element is 5 at index 3
IndexOfMin<T>

Gets the index of the minimum element in the list.

Parameters:

  • list (IList<T>): The list to find the minimum element in.

Returns:

  • int: The index of the minimum element in the list.

Example Usage:

var numbers = new List<int> { 1, 3, 2, 5, 4 };
int minIndex = numbers.IndexOfMin();
// minIndex is now 0, as the minimum element is 1 at index 0
RemoveDuplicates<T>

Removes duplicates from the list while preserving order.

Parameters:

  • list (IList<T>): The list to remove duplicates from.

Example Usage:

var numbers = new List<int> { 1, 2, 2, 3, 4, 4, 5 };
numbers.RemoveDuplicates();
// numbers is now { 1, 2, 3, 4, 5 }

String Extensions

The StringExtensions class provides several useful extension methods for working with strings.

Methods

Minify

Minifies a text by replacing spaces, tabs, and line breaks with a single space.

Parameters:

  • bigText (string): The text to be minified.

Returns:

  • string: The minified text.

Example Usage:

string text = @"This is a   test.

                New line.";
string minifiedText = text.Minify();
Console.WriteLine(minifiedText); // Output: "This is a test. New line."
StripAccents

Removes all diacritics (accents) from a string.

Parameters:

  • text (string): The text from which to remove diacritics.

Returns:

  • string: The text without diacritics.

Example Usage:

string text = "Café";
string strippedText = text.StripAccents();
Console.WriteLine(strippedText); // Output: "Cafe"
OnlyDigits

Removes everything that is not a digit from a string.

Parameters:

  • text (string): The target string.

Returns:

  • string: A string containing only digits.

Example Usage:

string text = "Phone: 123-456-7890";
string digits = text.OnlyDigits();
Console.WriteLine(digits); // Output: "1234567890"

Operation Result

The Result<TPayload> class represents the result of an operation, which can either be a success with a payload or a failure with an error. This class provides methods to map and process the result based on its success or failure state.

It's a somewhat functional approach to handling operation results, not fully adhering to the functional programming paradigm but providing some of its benefits.

Example Usage

using Raccoon.Ninja.Tools.OperationResult;
using Raccoon.Ninja.Tools.OperationResult.ResultError;

class Program
{
    static void Main()
    {
        // Example mapping result.
        var result = DoSomething();
        result.Map(
            success => Console.WriteLine($"Success! Here's the result: {success}"), // Only executed if successful
            error => Console.WriteLine($"Oh no! It failed! Error: {error.ErrorMessage}") // Only executed if failed
        );

        // Example processing result and getting an object back.
        var result2 = DoSomethingElseAndReturnPayload();
        var processedPayload = failureResult.Process(
            success => $"Processed: {success}", // Only executed if successful
            failure => $"Failed: {failure.ErrorMessage}" // Only executed if failed
        );
        Console.WriteLine(processedPayload);
    }
}

Performance

Quick update here. I caved in and changed from class to readonly struct. I think the performance benefits, thread safety, and immutability are worth it. Thankfully, this won't affect anyone, because the contracts are the same.

Below are the results of the benchmark tests for instantiating Classes, Struct, Readonly Struct, and Records. I'm aware that they are not the best, but it's good enough for a brief comparison.

Method Mean Error StdDev Median Rank Gen0 Allocated
NewClass 5.7458 ns 0.2447 ns 0.7216 ns 5.6876 ns 2 0.0051 32 B
NewStruct 0.0688 ns 0.0320 ns 0.0938 ns 0.0008 ns 1 - -
NewReadonlyStruct 0.0795 ns 0.0341 ns 0.0990 ns 0.0370 ns 1 - -
NewRecord 5.6249 ns 0.2198 ns 0.6482 ns 5.5565 ns 2 0.0051 32 B

To no one's surprise, instantiating a struct or readonly struct is by far the fastest option and won't allocate any memory.

Other

Shamelessly plugging the link to my site: https://raccoon.ninja

Product Compatible and additional computed target framework versions.
.NET net8.0 is compatible.  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. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • net8.0

    • No dependencies.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.8.0 78 9/8/2024
1.7.0 85 9/7/2024
1.6.0 79 9/7/2024
1.5.0 82 9/7/2024
1.4.0 83 9/2/2024
1.3.0 84 9/1/2024
1.2.0 85 9/1/2024
1.1.0 88 8/31/2024

Added the following ListExtension methods:
 - `Shuffle`: This method shuffles the elements of a list in place;
 - `Random`: This method returns a random element from a list;
 - `PopLast`: This method removes and returns the last element of a list;
 - `PopFirst`: This method removes and returns the first element of a list;
 - `IndexOfMax`: This method returns the index of the maximum element in a list;
 - `IndexOfMin`: This method returns the index of the minimum element in a list;
 - `RemoveDuplicates`: This method removes all duplicates from a list;