Compare commits

...

8 Commits
v0.2.0 ... main

Author SHA1 Message Date
Lexi / Zoe b2c7df32eb
Fix username validation in account edit
continuous-integration/drone/push Build is passing Details
continuous-integration/drone/tag Build is passing Details
2022-01-04 01:33:51 +01:00
Lexi / Zoe bf8a87ff30
Drone CI: Ignore PHP version from composer image
continuous-integration/drone/push Build is passing Details
continuous-integration/drone/tag Build is passing Details
2022-01-02 03:31:18 +01:00
Lexi / Zoe 2f20e9d474
Sort aliases by wildcard priority first
continuous-integration/drone/push Build is failing Details
continuous-integration/drone Build is failing Details
2022-01-02 03:06:14 +01:00
Lexi / Zoe 6400962a44
Implement wildcard aliases 2022-01-02 03:05:40 +01:00
Lexi / Zoe a01ddd9b22
Include unannotated tags in "make version"
continuous-integration/drone/push Build encountered an error Details
continuous-integration/drone/tag Build encountered an error Details
2021-12-29 18:45:57 +01:00
Lexi / Zoe d2fe2e9a8d
Fix max length string validation in FormData
continuous-integration/drone/push Build encountered an error Details
continuous-integration/drone/tag Build encountered an error Details
2021-12-29 18:36:02 +01:00
Lexi / Zoe bdd0f520b4
Fix "undefined index" in YamlConfigLoader for casted values 2021-10-30 18:49:22 +02:00
Lexi / Zoe 68070a614d
Fix phony version target in Makefile
continuous-integration/drone/push Build encountered an error Details
2021-09-26 00:50:23 +02:00
12 changed files with 149 additions and 33 deletions

View File

@ -12,7 +12,7 @@ steps:
environment: environment:
COMPOSER_CACHE_DIR: /tmp/cache COMPOSER_CACHE_DIR: /tmp/cache
commands: commands:
- composer install --no-progress --no-interaction - composer install --no-progress --no-interaction --ignore-platform-reqs
- name: run unit tests - name: run unit tests
image: php:7.4 image: php:7.4

View File

@ -9,7 +9,7 @@ DOCKER_RUN = $(DOCKER_COMPOSE) run --rm app
COMPOSER = $(DOCKER_RUN) composer COMPOSER = $(DOCKER_RUN) composer
PHPUNIT = $(DOCKER_RUN) vendor/bin/phpunit PHPUNIT = $(DOCKER_RUN) vendor/bin/phpunit
.PHONY: all clean VERSION \ .PHONY: all clean version \
docker-up docker-up-detached docker-down docker-restart docker-build docker-rebuild docker-purge docker-logs docker-run \ docker-up docker-up-detached docker-down docker-restart docker-build docker-rebuild docker-purge docker-logs docker-run \
composer-install composer-install-no-dev composer-update composer-cmd \ composer-install composer-install-no-dev composer-update composer-cmd \
test phpunit open-coverage test phpunit open-coverage
@ -99,7 +99,7 @@ open-coverage:
# Create VERSION file from current git tag # Create VERSION file from current git tag
version: version:
git describe | tee VERSION git describe --tags | tee VERSION
# Clean up # Clean up

View File

@ -27,6 +27,10 @@ a:hover, a:focus {
text-decoration: underline; text-decoration: underline;
} }
.monospace {
font-family: monospace;
}
.gray { .gray {
color: gray; color: gray;
} }
@ -248,7 +252,6 @@ details > summary {
details > p { details > p {
margin: 0.75rem 1rem; margin: 0.75rem 1rem;
font-family: monospace;
} }
/* -- Detail columns -- */ /* -- Detail columns -- */

View File

@ -38,6 +38,7 @@ CREATE TABLE `mail_aliases`
`alias_id` int(10) unsigned NOT NULL AUTO_INCREMENT, `alias_id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`user_id` int(10) unsigned NOT NULL, `user_id` int(10) unsigned NOT NULL,
`mail_address` varchar(255) NOT NULL, `mail_address` varchar(255) NOT NULL,
`wildcard_priority` smallint(6) NOT NULL DEFAULT 0,
`created_at` datetime NOT NULL DEFAULT NOW(), `created_at` datetime NOT NULL DEFAULT NOW(),
`modified_at` datetime NOT NULL DEFAULT NOW(), `modified_at` datetime NOT NULL DEFAULT NOW(),
PRIMARY KEY (`alias_id`), PRIMARY KEY (`alias_id`),

View File

@ -20,7 +20,7 @@ abstract class FormData
throw new InputValidationError("$fieldName is required."); throw new InputValidationError("$fieldName is required.");
} elseif (strlen($raw) < $minLength) { } elseif (strlen($raw) < $minLength) {
throw new InputValidationError("$fieldName is too short (minimum $minLength characters)."); throw new InputValidationError("$fieldName is too short (minimum $minLength characters).");
} elseif (strlen($raw) > 100) { } elseif (strlen($raw) > $maxLength) {
throw new InputValidationError("$fieldName is too long (maximum $maxLength characters)."); throw new InputValidationError("$fieldName is too long (maximum $maxLength characters).");
} }
return $raw; return $raw;
@ -44,25 +44,42 @@ abstract class FormData
// Input validation - Application specific validators // Input validation - Application specific validators
protected static function validateUsername(string $username, bool $required = true, string $fieldName = 'Username'): ?string private static function validateMailAddress(string $address, bool $required = true, string $fieldName = 'Mail address'): ?string
{ {
if (!$required && $username === '') { // Note: This validator allows '%' as wildcard character inside mail addresses.
if (!$required && $address === '') {
return null; return null;
} }
$username = strtolower( $address = strtolower(
self::validateString($username, 3, 100, $fieldName) self::validateString($address, 3, 100, $fieldName)
); );
if (!preg_match('/^[a-z0-9._+-]+@[a-z0-9.-]+$/', $username) || preg_match('/^\\.|\\.\\.|\\.@|@\\.|\\.$/', $username)) { if (!preg_match('/^[a-z0-9%._+-]+@[a-z0-9%.-]+$/', $address) || preg_match('/^\\.|\\.\\.|\\.@|@\\.|\\.$/', $address)) {
throw new InputValidationError("$fieldName is not valid (must be a valid mail address)."); throw new InputValidationError("$fieldName is not valid (must be a valid mail address).");
} }
return $address;
}
protected static function validateUsername(string $username, bool $required = true): ?string
{
$username = self::validateMailAddress($username, $required, 'Username');
if ($username !== null && strpos($username, '%') !== false) {
throw new InputValidationError('Username must not contain the wildcard character "%" (use a wildcard alias instead).');
}
return $username; return $username;
} }
protected static function validateAliasAddress(string $aliasAddress): string protected static function validateAliasAddress(string $aliasAddress, bool $isWildcard): string
{ {
return self::validateUsername($aliasAddress, true, 'Alias address'); $aliasAddress = self::validateMailAddress($aliasAddress, true, 'Alias address');
// Check if the address contains a wildcard character
if (!$isWildcard && strpos($aliasAddress, '%') !== false) {
throw new InputValidationError('Non-wildcard alias address must not contain "%" character.');
}
return $aliasAddress;
} }
protected static function validatePassword(string $password, string $passwordRepeat, bool $required = true): ?string protected static function validatePassword(string $password, string $passwordRepeat, bool $required = true): ?string

View File

@ -30,7 +30,7 @@ class YamlConfigLoader implements ConfigLoaderInterface
// App settings // App settings
'appTitle' => $parsedConfig['appTitle'] ?? null, 'appTitle' => $parsedConfig['appTitle'] ?? null,
'environment' => $parsedConfig['environment'] ?? null, 'environment' => $parsedConfig['environment'] ?? null,
'debug' => (bool)$parsedConfig['debug'] ?? null, 'debug' => isset($parsedConfig['debug']) ? (bool)$parsedConfig['debug'] : null,
'timezone' => $parsedConfig['timezone'] ?? null, 'timezone' => $parsedConfig['timezone'] ?? null,
'dateTimeFormat' => $parsedConfig['dateTimeFormat'] ?? null, 'dateTimeFormat' => $parsedConfig['dateTimeFormat'] ?? null,
@ -39,7 +39,7 @@ class YamlConfigLoader implements ConfigLoaderInterface
// Database settings // Database settings
'databaseHost' => $parsedConfig['database']['host'] ?? null, 'databaseHost' => $parsedConfig['database']['host'] ?? null,
'databasePort' => (int)$parsedConfig['database']['port'] ?? null, 'databasePort' => isset($parsedConfig['database']['port']) ? (int)$parsedConfig['database']['port'] : null,
'databaseName' => $parsedConfig['database']['name'] ?? null, 'databaseName' => $parsedConfig['database']['name'] ?? null,
'databaseUsername' => $parsedConfig['database']['username'] ?? null, 'databaseUsername' => $parsedConfig['database']['username'] ?? null,
'databasePassword' => $parsedConfig['database']['password'] ?? null, 'databasePassword' => $parsedConfig['database']['password'] ?? null,

View File

@ -4,20 +4,34 @@ declare(strict_types=1);
namespace MailAccountAdmin\Frontend\Accounts; namespace MailAccountAdmin\Frontend\Accounts;
use MailAccountAdmin\Common\FormData; use MailAccountAdmin\Common\FormData;
use MailAccountAdmin\Exceptions\InputValidationError;
class AccountAddAliasData extends FormData class AccountAddAliasData extends FormData
{ {
private string $aliasAddress; private string $aliasAddress;
private bool $isWildcard;
private int $wildcardPriority;
private function __construct(string $aliasAddress) private function __construct(string $aliasAddress, bool $isWildcard, int $wildcardPriority)
{ {
if ($isWildcard && $wildcardPriority === 0) {
throw new InputValidationError('Wildcard alias must have a wildcard priority other than 0.');
} elseif (!$isWildcard) {
$wildcardPriority = 0;
}
$this->aliasAddress = $aliasAddress; $this->aliasAddress = $aliasAddress;
$this->isWildcard = $isWildcard;
$this->wildcardPriority = $wildcardPriority;
} }
public static function createFromArray(array $raw): self public static function createFromArray(array $raw): self
{ {
$isWildcard = self::validateBoolOption(trim($raw['is_wildcard'] ?? ''));
return new self( return new self(
self::validateAliasAddress(trim($raw['alias_address'] ?? '')), self::validateAliasAddress(trim($raw['alias_address'] ?? ''), $isWildcard),
$isWildcard,
self::validateInteger(trim($raw['wildcard_priority'] ?? '')),
); );
} }
@ -25,4 +39,14 @@ class AccountAddAliasData extends FormData
{ {
return $this->aliasAddress; return $this->aliasAddress;
} }
public function isWildcard(): bool
{
return $this->isWildcard;
}
public function getWildcardPriority(): int
{
return $this->wildcardPriority;
}
} }

View File

@ -278,14 +278,22 @@ class AccountHandler
// Check if account exists // Check if account exists
$this->ensureAccountExists($accountId); $this->ensureAccountExists($accountId);
$unescapedAddress = $address = $aliasAddData->getAliasAddress();
$wildcardPriority = 0;
if ($aliasAddData->isWildcard()) {
// If it's a wildcard alias, escape underscores
$address = str_replace('_', '\\_', $address);
$wildcardPriority = $aliasAddData->getWildcardPriority();
}
// Check if alias address is still available // Check if alias address is still available
$address = $aliasAddData->getAliasAddress();
if (!$this->accountRepository->checkUsernameAvailable($address) || !$this->aliasRepository->checkAliasAvailable($address)) { if (!$this->accountRepository->checkUsernameAvailable($address) || !$this->aliasRepository->checkAliasAvailable($address)) {
throw new InputValidationError("Alias address \"$address\" is not available."); throw new InputValidationError("Alias address \"$unescapedAddress\" is not available.");
} }
// Create alias in database // Create alias in database
$this->aliasRepository->createNewAlias($accountId, $address); $this->aliasRepository->createNewAlias($accountId, $address, $wildcardPriority);
} }

View File

@ -10,14 +10,23 @@ class Alias
private int $id; private int $id;
private int $userId; private int $userId;
private string $mailAddress; private string $mailAddress;
private int $wildcardPriority;
private DateTimeImmutable $createdAt; private DateTimeImmutable $createdAt;
private DateTimeImmutable $modifiedAt; private DateTimeImmutable $modifiedAt;
private function __construct(int $id, int $userId, string $mailAddress, DateTimeImmutable $createdAt, DateTimeImmutable $modifiedAt) private function __construct(
int $id,
int $userId,
string $mailAddress,
int $wildcardPriority,
DateTimeImmutable $createdAt,
DateTimeImmutable $modifiedAt
)
{ {
$this->id = $id; $this->id = $id;
$this->userId = $userId; $this->userId = $userId;
$this->mailAddress = $mailAddress; $this->mailAddress = $mailAddress;
$this->wildcardPriority = $wildcardPriority;
$this->createdAt = $createdAt; $this->createdAt = $createdAt;
$this->modifiedAt = $modifiedAt; $this->modifiedAt = $modifiedAt;
} }
@ -28,6 +37,7 @@ class Alias
(int)$data['alias_id'], (int)$data['alias_id'],
(int)$data['user_id'], (int)$data['user_id'],
$data['mail_address'], $data['mail_address'],
(int)$data['wildcard_priority'],
new DateTimeImmutable($data['created_at']), new DateTimeImmutable($data['created_at']),
new DateTimeImmutable($data['modified_at']), new DateTimeImmutable($data['modified_at']),
); );
@ -43,11 +53,24 @@ class Alias
return $this->userId; return $this->userId;
} }
public function getMailAddress(): string public function getMailAddress(bool $unescape = true): string
{ {
if ($this->isWildcard() && $unescape) {
return str_replace('\\_', '_', $this->mailAddress);
}
return $this->mailAddress; return $this->mailAddress;
} }
public function isWildcard(): bool
{
return $this->wildcardPriority !== 0;
}
public function getWildcardPriority(): int
{
return $this->wildcardPriority;
}
public function getCreatedAt(): DateTimeImmutable public function getCreatedAt(): DateTimeImmutable
{ {
return $this->createdAt; return $this->createdAt;

View File

@ -11,11 +11,11 @@ class AliasRepository extends BaseRepository
{ {
public function fetchAliasesForUserId(int $userId): array public function fetchAliasesForUserId(int $userId): array
{ {
$statement = $this->pdo->prepare('SELECT * FROM mail_aliases WHERE user_id = :user_id ORDER BY mail_address'); $statement = $this->pdo->prepare('SELECT * FROM mail_aliases WHERE user_id = :user_id ORDER BY wildcard_priority, mail_address');
$statement->execute(['user_id' => $userId]); $statement->execute(['user_id' => $userId]);
$rows = $statement->fetchAll(PDO::FETCH_ASSOC); $rows = $statement->fetchAll(PDO::FETCH_ASSOC);
// Create Account models from rows // Create Alias models from rows
$aliasList = []; $aliasList = [];
foreach ($rows as $row) { foreach ($rows as $row) {
$aliasList[] = Alias::createFromArray($row); $aliasList[] = Alias::createFromArray($row);
@ -43,12 +43,20 @@ class AliasRepository extends BaseRepository
return $statement->rowCount() === 0; return $statement->rowCount() === 0;
} }
public function createNewAlias(int $userId, string $mailAddress): void public function createNewAlias(int $userId, string $mailAddress, int $wildcardPriority = 0): void
{ {
$statement = $this->pdo->prepare('INSERT INTO mail_aliases (user_id, mail_address) VALUES (:user_id, :mail_address)'); $query = '
INSERT INTO mail_aliases
(user_id, mail_address, wildcard_priority)
VALUES
(:user_id, :mail_address, :wildcard_priority)
';
$statement = $this->pdo->prepare($query);
$statement->execute([ $statement->execute([
'user_id' => $userId, 'user_id' => $userId,
'mail_address' => $mailAddress, 'mail_address' => $mailAddress,
'wildcard_priority' => $wildcardPriority,
]); ]);
} }

View File

@ -24,7 +24,7 @@
</p> </p>
<details> <details>
<summary>Show list of known domains</summary> <summary>Show list of known domains</summary>
<p>{{ domainList ? domainList | join(', ') : 'No domains exist yet.' }}</p> <p class="monospace">{{ domainList ? domainList | join(', ') : 'No domains exist yet.' }}</p>
</details> </details>
<table> <table>
<tr> <tr>

View File

@ -71,13 +71,21 @@
<table class="bordered_table"> <table class="bordered_table">
<tr> <tr>
<th></th> <th></th>
<th>Address</th> <th style="min-width: 16em">Address</th>
<th>Wildcard (priority)</th>
<th>Created at</th> <th>Created at</th>
</tr> </tr>
{% for alias in aliases %} {% for alias in aliases %}
<tr> <tr>
<td><input type="checkbox" id="delete_selected_{{ alias.getId() }}" name="selected_aliases[]" value="{{ alias.getId() }}"/></td> <td><input type="checkbox" id="delete_selected_{{ alias.getId() }}" name="selected_aliases[]" value="{{ alias.getId() }}"/></td>
<td><label for="delete_selected_{{ alias.getId() }}">{{ alias.getMailAddress() }}</label></td> <td><label for="delete_selected_{{ alias.getId() }}">{{ alias.getMailAddress() }}</label></td>
<td>
{% if alias.isWildcard() %}
Yes ({{ alias.getWildcardPriority() }})
{% else %}
<span class="gray">No</span>
{% endif %}
</td>
<td>{{ alias.getCreatedAt() | date }}</td> <td>{{ alias.getCreatedAt() | date }}</td>
</tr> </tr>
{% endfor %} {% endfor %}
@ -95,7 +103,18 @@
<tr> <tr>
<td><label for="add_alias_address">New alias:</label></td> <td><label for="add_alias_address">New alias:</label></td>
<td> <td>
<input id="add_alias_address" name="alias_address" value="{{ formData['new_alias'] | default('') }}" {{ success is defined or error is defined ? 'autofocus': '' }}/> <input id="add_alias_address" name="alias_address" value="{{ formData['alias_address'] | default('') }}" {{ success is defined or error is defined ? 'autofocus': '' }}/>
</td>
</tr>
<tr>
<td style="vertical-align: top">Wildcard:</td>
<td style="vertical-align: top">
<label>
<input type="checkbox" name="is_wildcard" {{ formData is defined and formData['is_wildcard'] | default() ? 'checked' : '' }}/>
Create wildcard alias,
</label>
<label for="add_alias_wildcard_priority">priority:</label>
<input type="number" id="add_alias_wildcard_priority" name="wildcard_priority" value="{{ formData['wildcard_priority'] | default('1') }}" style="width: 6em"/>
</td> </td>
</tr> </tr>
<tr> <tr>
@ -104,5 +123,18 @@
</td> </td>
</tr> </tr>
</table> </table>
<details>
<summary>How to define wildcard aliases?</summary>
<p>
Wildcard aliases use <code>%</code> as a placeholder for any amount of characters (zero or more),
e.g. <code>%@example.com</code> or <code>example-%@example.com</code>.
</p>
<p>
Additionally, a <b>wildcard priority</b> must be specified. When a mail address is looked up that
matches multiple (wildcard) aliases, the alias with the <b>lowest</b> wildcard priority will be used.
The priority must not be 0 (internally, the value 0 stands for regular non-wildcard aliases).
</p>
</details>
</form> </form>
{% endblock %} {% endblock %}