Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/chucker/passwordrulessharp

A C# implementation of an Apple-compatible Password Rules syntax
https://github.com/chucker/passwordrulessharp

Last synced: 2 months ago
JSON representation

A C# implementation of an Apple-compatible Password Rules syntax

Awesome Lists containing this project

README

        

# PasswordRulesSharp
A C# implementation of an Apple-compatible Password Rules syntax

## Usage

You probably want to install it from NuGet: https://www.nuget.org/packages/PasswordRulesSharp/2.0.0.

### Building a rule

You an build a rule either from the raw syntax, or with a builder.

#### From raw syntax

You can use
[Apple's tool](https://developer.apple.com/password-rules/) to help
create a rule.

For example, let's use
`minlength: 20; required: lower; required: upper; required: digit; required: [-];`.
This requires that the password is at least 20 characters wrong, has at least
one lower- and uppercase character each, has a digit, and has a hyphen (`-`).

To instantiate this as a rule, you use `Rule.FromString()`:

```csharp
const string rawRule = "minlength: 20; required: lower; required: upper; required: digit; required: [-]";

var rule = Rule.FromString(rawRule);
```

#### Using the builder

You can create the same rule as above with `RuleBuilder`:

```csharp
var rule = new RuleBuilder()
.MinLength(20)
.RequireLower()
.RequireUpper()
.RequireDigit()
.RequireAnyOf("-")
.Build();
```

### Validation

With either of the two approaches to assigning `rule`, you can now take a
password and validate whether it matches that rule:

```csharp
const string samplePassword1 = "Hello2AndSomeMoreText-";

var validator = new PasswordRulesSharp.Validator.Validator(rule);

bool password1IsValid = validator.PasswordIsValid(samplePassword1, out var results1);
```

This returns `true`, as the password matches all rules!

Let's try with one that doesn't:

```csharp
const string samplePassword2 = "Hello1";

bool password2IsValid = validator.PasswordIsValid(samplePassword2, out var results2);
```

This returns `false`, of course. You can look at `results2` as to why that is:
the library found five requirements. Four
are `CharacterClassRequirement`s, and the fifth is a
`MinimumLengthRequirement`. The sample password fulfills three of the character
class requirements, but it doesn't have a hyphen, so the validator fails that
one. Moreover, the password also doesn't meet the minimum length.

This collection of detailed validation failure results is useful if you want to
interactively tell the user why their proposed password cannot be used. But if
you want prefer something simpler, you can also discard them, and simply look
at the boolean result:

```csharp
bool password2IsValid = validator.PasswordIsValid(samplePassword2, out _);
```

This is still enough if you just want a yes/no "is it valid" answer.

### Generation

This flow is enough if your users want to pick passwords themselves. But you may also prefer for passwords to be
_generated_. For that, instantiate the `Generator`:

```csharp
var generator = new PasswordRulesSharp.Generator.Generator(rule);

var newPassword = generator.GeneratePassword();
```

You may also find it useful to generate a handful of passwords:

```csharp
for (int i = 0; i < 5; i++)
{
Console.WriteLine(generator.GeneratePassword());
}
```

The output will look like:

-0fBAp4q24d-VA9tO2wC
-nA7A6q-o9S4Jo-1SkD1
80piB-N4Mv8q-AxG0-o8
-hY0i6T-5UcI6jNQ7q-f
b-a7B6C-2FuO8rZyv4R-

All generated passwords will match the rules we've specified. As you can see,
they're all 20 characters long, and each contain at least one hyphen, one
digit, lower-case character, and upper-case character.

#### Of note:

* If you pass a `minlength` _below_ `20`, the generated passwords will still be
at least 20 characters long, for security reasons. If you _want_ to generate a
shorter password, you can achieve that by also specifying a `maxlength`. For
example, `minlength: 8; maxlength: 10; required: lower; required: upper; required: digit; required: [-]`
will yield generated passwords that are 10 characters long.

* As you can see, there is no logic for the generated passwords to
follow a particular pattern. This increases entropy (and therefore security),
but "looks" slightly less nice.

## References

Apple's docs: https://developer.apple.com/documentation/security/password_autofill/customizing_password_autofill_rules

Apple's validation tool: https://developer.apple.com/password-rules/

A Rust port: https://docs.rs/password-rules-parser/latest/password_rules_parser/

### Further reading

There's also https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.identity.passwordoptions, but it is far more limiting. (Someone might write an importer from that format.)