backend/source/repositories/member.ts

368 lines
8.1 KiB
TypeScript
Raw Normal View History

2024-05-20 21:56:48 +02:00
/*
Espe | Ein schlichtes Werkzeug zur Mitglieder-Verwaltung | Backend
Copyright (C) 2024 Christian Fraß
This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public
License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later
version.
This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with this program. If not, see
<https://www.gnu.org/licenses/>.
*/
2024-04-29 20:48:20 +02:00
namespace _espe.repository.member
{
2025-01-17 09:49:13 +01:00
/**
*/
type type_group_chest = lib_plankton.storage.type_chest<
Array<any>,
Record<string, any>,
lib_plankton.database.type_description_create_table,
lib_plankton.storage.sql_table_common.type_sql_table_common_search_term,
Record<string, any>
>;
2024-04-29 20:48:20 +02:00
/**
*/
2025-01-17 09:49:13 +01:00
var _core_store : (
2024-04-29 20:48:20 +02:00
null
|
lib_plankton.storage.type_store<
2024-04-29 22:40:52 +02:00
_espe.type.member_id,
2024-04-29 20:48:20 +02:00
Record<string, any>,
{},
lib_plankton.storage.type_sql_table_autokey_search_term,
Record<string, any>
>
) = null;
/**
*/
2025-01-17 09:49:13 +01:00
var _group_chest : (
null
|
type_group_chest
) = null;
/**
*/
function get_core_store(
2024-04-29 20:48:20 +02:00
) : lib_plankton.storage.type_store<
2024-04-29 22:40:52 +02:00
_espe.type.member_id,
2024-04-29 20:48:20 +02:00
Record<string, any>,
{},
lib_plankton.storage.type_sql_table_autokey_search_term,
Record<string, any>
>
{
2025-01-17 09:49:13 +01:00
if (_core_store === null) {
_core_store = lib_plankton.storage.sql_table_autokey_store(
2024-04-29 20:48:20 +02:00
{
"database_implementation": _espe.helpers.database_implementation(),
"table_name": "members",
"key_name": "id",
}
);
}
else {
// do nothing
}
2025-01-17 09:49:13 +01:00
return _core_store;
2024-04-29 20:48:20 +02:00
}
/**
*/
2025-01-17 09:49:13 +01:00
function get_group_chest(
) : type_group_chest
2024-04-29 20:48:20 +02:00
{
2025-01-17 09:49:13 +01:00
if (_group_chest === null) {
_group_chest = lib_plankton.storage.sql_table_common.chest(
{
"database_implementation": _espe.helpers.database_implementation(),
"table_name": "member_groups",
"key_names": ["member_id","group"],
}
);
}
else {
// do nothing
}
return _group_chest;
2024-04-29 20:48:20 +02:00
}
/**
*/
2025-01-17 09:49:13 +01:00
type type_dispersal = {
core_row : Record<string, any>;
group_rows : Array<
Record<string, any>
>;
};
/**
*/
function encode(
object : _espe.type.member_object
) : type_dispersal
2024-04-29 20:48:20 +02:00
{
return {
2025-01-17 09:49:13 +01:00
"core_row": {
"membership_number": object.membership_number,
"name_real_value": object.name_real_value,
"name_real_index": object.name_real_index,
"email_address_private": object.email_address_private,
"registered": (object.registered ? 1 : 0),
"enabled": (object.enabled ? 1 : 0),
"email_use_veiled_address": (object.email_use_veiled_address ? 1 : 0),
"email_use_nominal_address": (object.email_use_nominal_address ? 1 : 0),
"email_redirect_to_private_address": (object.email_redirect_to_private_address ? 1 : 0),
"email_allow_sending": (object.email_allow_sending ? 1 : 0),
"password_image": object.password_image,
"password_change_last_attempt": object.password_change_last_attempt,
"password_change_token": object.password_change_token,
},
"group_rows": (
object.groups
.map(
group => ({
"group": group,
})
)
)
2024-04-29 20:48:20 +02:00
};
}
/**
*/
2025-01-17 09:49:13 +01:00
function decode(
dispersal : type_dispersal
) : _espe.type.member_object
2024-04-29 20:48:20 +02:00
{
2025-01-17 09:49:13 +01:00
return {
"membership_number": dispersal.core_row["membership_number"],
"name_real_value": dispersal.core_row["name_real_value"],
"name_real_index": dispersal.core_row["name_real_index"],
"email_address_private": dispersal.core_row["email_address_private"],
"groups": lib_plankton.list.sorted<string>(
dispersal.group_rows.map(row => row["group"]),
(group1, group2) => ((group1 <= group2) ? 1 : 0)
),
"registered": (dispersal.core_row["registered"] > 0),
"enabled": (dispersal.core_row["enabled"] > 0),
"email_use_veiled_address": (dispersal.core_row["email_use_veiled_address"] > 0),
"email_use_nominal_address": (dispersal.core_row["email_use_nominal_address"] > 0),
"email_redirect_to_private_address": (dispersal.core_row["email_redirect_to_private_address"] > 0),
"email_allow_sending": (dispersal.core_row["email_allow_sending"] > 0),
"password_image": dispersal.core_row["password_image"],
"password_change_last_attempt": dispersal.core_row["password_change_last_attempt"],
"password_change_token": dispersal.core_row["password_change_token"],
};
2024-04-29 20:48:20 +02:00
}
/**
2024-04-30 01:32:24 +02:00
* @todo optimize
2024-04-29 20:48:20 +02:00
*/
export async function list(
2024-04-30 01:32:24 +02:00
search_term : (null | string)
2024-04-29 20:48:20 +02:00
) : Promise<
Array<
{
2024-04-29 22:40:52 +02:00
id : _espe.type.member_id;
2024-04-29 20:48:20 +02:00
preview : {
2024-04-30 01:32:24 +02:00
membership_number : string;
name_real_value : string;
name_real_index : int;
2024-04-29 20:48:20 +02:00
};
}
>
>
{
return (
2025-01-17 09:49:13 +01:00
(await get_core_store().search(null))
2024-04-30 01:32:24 +02:00
.filter(
({"key": key, "preview": preview}) => (
(
(search_term === null)
||
(search_term.length <= 1)
)
? true
: (
preview["membership_number"].toLowerCase().includes(search_term.toLowerCase())
||
preview["name_real_value"].toLowerCase().includes(search_term.toLowerCase())
)
)
)
2024-04-29 20:48:20 +02:00
.map(
({"key": key, "preview": preview}) => ({
"id": key,
"preview": {
"membership_number": preview["membership_number"],
2024-04-30 01:32:24 +02:00
"name_real_value": preview["name_real_value"],
"name_real_index": preview["name_real_index"],
2024-04-29 20:48:20 +02:00
}
})
)
);
}
/**
*/
export async function read(
2024-04-29 22:40:52 +02:00
id : _espe.type.member_id
) : Promise<_espe.type.member_object>
2024-04-29 20:48:20 +02:00
{
2025-01-17 09:49:13 +01:00
const core_row : Record<string, any> = await get_core_store().read(id);
const group_hits : Array<{key : Record<string, any>; preview : Record<string, any>;}> = await get_group_chest().search(
{
"expression": "member_id = $member_id",
"arguments": {"member_id": id}
}
);
const dispersal : type_dispersal = {
"core_row": core_row,
"group_rows": group_hits.map(
hit => ({
"group": hit.preview["group"]
})
),
};
2024-04-29 20:48:20 +02:00
2025-01-17 09:49:13 +01:00
return decode(dispersal);
2024-04-29 20:48:20 +02:00
}
/**
*/
export async function create(
2024-04-29 22:40:52 +02:00
value : _espe.type.member_object
) : Promise<_espe.type.member_id>
2024-04-29 20:48:20 +02:00
{
2025-01-17 09:49:13 +01:00
const dispersal : type_dispersal = encode(value);
const id : _espe.type.member_id = await get_core_store().create(dispersal.core_row);
for await (const group_row of dispersal.group_rows) {
await get_group_chest().write(
[
id,
group_row["group"],
],
{
"_dummy": null,
}
);
}
2024-04-29 20:48:20 +02:00
return id;
}
/**
2025-01-17 09:49:13 +01:00
* @todo replace groups smartly
2024-04-29 20:48:20 +02:00
*/
export async function update(
2024-04-29 22:40:52 +02:00
id : _espe.type.member_id,
value : _espe.type.member_object
2024-04-29 20:48:20 +02:00
) : Promise<void>
{
2025-01-17 09:49:13 +01:00
const dispersal : type_dispersal = encode(value);
2024-04-29 20:48:20 +02:00
2025-01-17 09:49:13 +01:00
// core
await get_core_store().update(id, dispersal.core_row);
// groups
const hits : Array<{key : Array<any>; preview : Record<string, any>;}> = await get_group_chest().search(
{
"expression": "member_id = $member_id",
"arguments": {"member_id": id}
}
);
for (const hit of hits) {
await get_group_chest().delete([hit.key]);
}
for await (const group_row of dispersal.group_rows) {
await get_group_chest().write(
[
id,
group_row["group"],
],
{
"_dummy": null,
}
);
}
2024-04-29 20:48:20 +02:00
}
/**
*/
2024-04-29 22:40:52 +02:00
export async function delete_(
id : _espe.type.member_id
2024-04-29 20:48:20 +02:00
) : Promise<void>
{
2025-01-17 09:49:13 +01:00
// groups
const hits : Array<{key : Array<any>; preview : Record<string, any>;}> = await get_group_chest().search(
{
"expression": "member_id = $member_id",
"arguments": {"member_id": id}
}
);
for (const hit of hits) {
await get_group_chest().delete([hit.key]);
}
// core
await get_core_store().delete(id);
}
/**
*/
export async function dump(
) : Promise<
Array<
{
id : _espe.type.member_id;
object : _espe.type.member_object;
}
>
>
{
return (
Promise.all(
(await get_core_store().search(null))
.map(hit => hit.key)
.map(
id => (
read(id)
.then(
(object) => ({
"id": id,
"object": object
})
)
)
)
)
);
2024-04-29 20:48:20 +02:00
}
}