1
0
mirror of https://github.com/hrfee/jfa-go.git synced 2025-01-08 17:30:11 +00:00

accounts: start advanced search filter support

uses the same format "<field>:<value>", but supports quoted <values>
(allows for spaces in them), and lays groundwork to support string and
date-type field filtering. Truthiness is supported, meaning you can
check if an email is set with "email:yes" for example.
This commit is contained in:
Harvey Tindall 2023-06-13 13:39:13 +01:00
parent bc4015ac50
commit 8a37663c89
Signed by: hrfee
GPG Key ID: BBC65952848FB1A2

View File

@ -775,40 +775,158 @@ export class accountsList {
} }
} }
search = (query: string): string[] => { search = (query: String): string[] => {
query = query.toLowerCase() console.log("called!");
let result: string[] = []; const queries: { [field: string]: { name: string, getter: string, bool: boolean, string: boolean, date: boolean }} = {
if (query.includes(":")) { // Support admin:<true/false> and disabled:<true/false> "admin": {
const words = query.split(" "); name: "Admin",
getter: "admin",
bool: true,
string: false,
date: false
},
"disabled": {
name: "Disabled",
getter: "disabled",
bool: true,
string: false,
date: false
},
"access-jfa": {
name: "Access jfa-go",
getter: "accounts_admin",
bool: true,
string: false,
date: false
},
"email": {
name: "Email",
getter: "email",
bool: true,
string: true,
date: false
},
"telegram": {
name: "Telegram",
getter: "telegram",
bool: true,
string: true,
date: false
},
"matrix": {
name: "Matrix",
getter: "matrix",
bool: true,
string: true,
date: false
},
"discord": {
name: "Discord",
getter: "discord",
bool: true,
string: true,
date: false
},
"expiry": {
name: "Expiry",
getter: "expiry",
bool: true,
string: false,
date: true
},
"last-active": {
name: "Last Active",
getter: "last_active",
bool: true,
string: false,
date: true
}
}
query = query.toLowerCase();
let result: string[] = [...this._ordering];
console.log("initial:", result);
if (!(query.includes(":"))) return result;
// const words = query.split(" ");
let words: string[] = [];
// FIXME: SPLIT BY SPACE, UNLESS IN QUOTES
let quoteSymbol = ``;
let queryStart = -1;
let lastQuote = -1;
for (let i = 0; i < query.length; i++) {
if (queryStart == -1 && query[i] != " " && query[i] != `"` && query[i] != `'`) {
queryStart = i;
}
if ((query[i] == `"` || query[i] == `'`) && (quoteSymbol == `` || query[i] == quoteSymbol)) {
if (lastQuote != -1) {
lastQuote = -1;
quoteSymbol = ``;
} else {
lastQuote = i;
quoteSymbol = query[i];
}
}
if (query[i] == " " || i == query.length-1) {
if (lastQuote != -1) {
continue;
} else {
words.push(query.substring(queryStart, i+1).replace(/['"]/g, ""));
queryStart = -1;
}
}
}
query = ""; query = "";
for (let word of words) { for (let word of words) {
if (word.includes(":")) { const split = word.split(":");
const querySplit = word.split(":")
let state = false; if (!(split[0] in queries)) continue;
if (querySplit[1] == "true" || querySplit[1] == "yes") {
state = true; const queryFormat = queries[split[0]];
if (queryFormat.bool) {
let isBool = false;
let boolState = false;
if (split[1] == "true" || split[1] == "yes" || split[1] == "t" || split[1] == "y") {
isBool = true;
boolState = true;
} else if (split[1] == "false" || split[1] == "no" || split[1] == "f" || split[1] == "n") {
isBool = true;
boolState = false;
} }
for (let id of this._ordering) { if (isBool) {
const user = this._users[id]; // console.log("is bool, state", boolState);
let attrib: boolean; // So removing elements doesn't affect us
if (querySplit[0] == "admin") { attrib = user.admin; } let cachedResult = [...result];
else if (querySplit[0] == "disabled") { attrib = user.disabled; } for (let id of cachedResult) {
if (attrib == state) { result.push(id); } const u = this._users[id];
} const value = Object.getOwnPropertyDescriptor(user.prototype, queryFormat.getter).get.call(u);
} else { query += word + " "; } // console.log("got", queryFormat.getter + ":", value);
// Remove from result if not matching query
if (!((value && boolState) || (!value && !boolState))) {
// console.log("not matching, result is", result);
result.splice(result.indexOf(id), 1);
} }
} }
if (query == "") { return result; } continue
for (let id of this._ordering) {
const user = this._users[id];
if (user.name.toLowerCase().includes(query)) {
result.push(id);
} else if (user.email.toLowerCase().includes(query)) {
result.push(id);
} }
} }
return result; if (queryFormat.string) {
// FIXME: STRING SEARCH FOR FIELD
// FIXME: SUBTRACT FROM RESULT
continue;
} }
if (queryFormat.date) {
}
}
return result
};
get selectAll(): boolean { return this._selectAll.checked; } get selectAll(): boolean { return this._selectAll.checked; }
set selectAll(state: boolean) { set selectAll(state: boolean) {