mirror of
https://code.mensbeam.com/MensBeam/Arsse.git
synced 2024-12-31 21:12:41 +00:00
WIP redesign of user properties
This commit is contained in:
parent
0f3e0411f0
commit
978929aabd
7 changed files with 142 additions and 96 deletions
119
lib/Database.php
119
lib/Database.php
|
@ -11,7 +11,7 @@ use JKingWeb\Arsse\Db\Statement;
|
||||||
use JKingWeb\Arsse\Misc\Query;
|
use JKingWeb\Arsse\Misc\Query;
|
||||||
use JKingWeb\Arsse\Context\Context;
|
use JKingWeb\Arsse\Context\Context;
|
||||||
use JKingWeb\Arsse\Misc\Date;
|
use JKingWeb\Arsse\Misc\Date;
|
||||||
use JKingWeb\Arsse\Misc\ValueInfo;
|
use JKingWeb\Arsse\Misc\ValueInfo as V;
|
||||||
use JKingWeb\Arsse\Misc\URL;
|
use JKingWeb\Arsse\Misc\URL;
|
||||||
|
|
||||||
/** The high-level interface with the database
|
/** The high-level interface with the database
|
||||||
|
@ -149,7 +149,7 @@ class Database {
|
||||||
$count = 0;
|
$count = 0;
|
||||||
$convType = Db\AbstractStatement::TYPE_NORM_MAP[Statement::TYPES[$type]];
|
$convType = Db\AbstractStatement::TYPE_NORM_MAP[Statement::TYPES[$type]];
|
||||||
foreach ($values as $v) {
|
foreach ($values as $v) {
|
||||||
$v = ValueInfo::normalize($v, $convType, null, "sql");
|
$v = V::normalize($v, $convType, null, "sql");
|
||||||
if (is_null($v)) {
|
if (is_null($v)) {
|
||||||
// nulls are pointless to have
|
// nulls are pointless to have
|
||||||
continue;
|
continue;
|
||||||
|
@ -161,7 +161,7 @@ class Database {
|
||||||
$clause[] = $this->db->literalString($v);
|
$clause[] = $this->db->literalString($v);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
$clause[] = ValueInfo::normalize($v, ValueInfo::T_STRING, null, "sql");
|
$clause[] = V::normalize($v, V::T_STRING, null, "sql");
|
||||||
}
|
}
|
||||||
$count++;
|
$count++;
|
||||||
}
|
}
|
||||||
|
@ -299,32 +299,43 @@ class Database {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function userPropertiesGet(string $user): array {
|
public function userPropertiesGet(string $user, bool $includeLarge = true): array {
|
||||||
$out = $this->db->prepare("SELECT num, admin, lang, tz, sort_asc from arsse_users where id = ?", "str")->run($user)->getRow();
|
$meta = $this->db->prepareArray(
|
||||||
if (!$out) {
|
"SELECT \"key\", value from arsse_user_meta where owner = ? and \"key\" not in ('num', 'admin')
|
||||||
|
union all select 'num', num from arsse_users where id = ?
|
||||||
|
union all select 'admin', admin from arsse_users where id = ?",
|
||||||
|
["str", "str", "str"]
|
||||||
|
)->run($user)->getRow();
|
||||||
|
if (!$meta) {
|
||||||
throw new User\ExceptionConflict("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
throw new User\ExceptionConflict("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
||||||
}
|
}
|
||||||
settype($out['num'], "int");
|
$meta = array_combine(array_column($meta, "key"), array_column($meta, "value"));
|
||||||
settype($out['admin'], "bool");
|
settype($meta['num'], "integer");
|
||||||
settype($out['sort_asc'], "bool");
|
return $meta;
|
||||||
return $out;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function userPropertiesSet(string $user, array $data): bool {
|
public function userPropertiesSet(string $user, array $data): bool {
|
||||||
if (!$this->userExists($user)) {
|
if (!$this->userExists($user)) {
|
||||||
throw new User\ExceptionConflict("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
throw new User\ExceptionConflict("doesNotExist", ["action" => __FUNCTION__, "user" => $user]);
|
||||||
}
|
}
|
||||||
$allowed = [
|
$update = $this->db->prepare("UPDATE arsse_user_meta set value = ? where owner = ? and \"key\" = ?", "str", "str", "str");
|
||||||
'admin' => "strict bool",
|
$insert = ["INSERT INTO arsse_user_meta values(?, ?, ?)", "str", "strict str", "str"];
|
||||||
'lang' => "str",
|
foreach ($data as $k => $v) {
|
||||||
'tz' => "strict str",
|
if ($k === "admin") {
|
||||||
'sort_asc' => "strict bool",
|
$this->db->prepare("UPDATE arsse_users SET admin = ? where user = ?", "bool", "str")->run($v, $user);
|
||||||
];
|
} elseif ($k === "num") {
|
||||||
[$setClause, $setTypes, $setValues] = $this->generateSet($data, $allowed);
|
continue;
|
||||||
if (!$setClause) {
|
} else {
|
||||||
return false;
|
$success = $update->run($v, $user, $k)->changes();
|
||||||
|
if (!$success) {
|
||||||
|
if (!$insert instanceof Db\Statement) {
|
||||||
|
$insert = $this->db->prepare(...$insert);
|
||||||
}
|
}
|
||||||
return (bool) $this->db->prepare("UPDATE arsse_users set $setClause where id = ?", $setTypes, "str")->run($setValues, $user)->changes();
|
$insert->run($user, $k, $v);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Creates a new session for the given user and returns the session identifier */
|
/** Creates a new session for the given user and returns the session identifier */
|
||||||
|
@ -515,7 +526,7 @@ class Database {
|
||||||
* @param integer $id The identifier of the folder to delete
|
* @param integer $id The identifier of the folder to delete
|
||||||
*/
|
*/
|
||||||
public function folderRemove(string $user, $id): bool {
|
public function folderRemove(string $user, $id): bool {
|
||||||
if (!ValueInfo::id($id)) {
|
if (!V::id($id)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "folder", 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "folder", 'type' => "int > 0"]);
|
||||||
}
|
}
|
||||||
$changes = $this->db->prepare("DELETE FROM arsse_folders where owner = ? and id = ?", "str", "int")->run($user, $id)->changes();
|
$changes = $this->db->prepare("DELETE FROM arsse_folders where owner = ? and id = ?", "str", "int")->run($user, $id)->changes();
|
||||||
|
@ -527,7 +538,7 @@ class Database {
|
||||||
|
|
||||||
/** Returns the identifier, name, and parent of the given folder as an associative array */
|
/** Returns the identifier, name, and parent of the given folder as an associative array */
|
||||||
public function folderPropertiesGet(string $user, $id): array {
|
public function folderPropertiesGet(string $user, $id): array {
|
||||||
if (!ValueInfo::id($id)) {
|
if (!V::id($id)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "folder", 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "folder", 'type' => "int > 0"]);
|
||||||
}
|
}
|
||||||
$props = $this->db->prepare("SELECT id,name,parent from arsse_folders where owner = ? and id = ?", "str", "int")->run($user, $id)->getRow();
|
$props = $this->db->prepare("SELECT id,name,parent from arsse_folders where owner = ? and id = ?", "str", "int")->run($user, $id)->getRow();
|
||||||
|
@ -593,7 +604,7 @@ class Database {
|
||||||
*/
|
*/
|
||||||
protected function folderValidateId(string $user, $id = null, bool $subject = false): array {
|
protected function folderValidateId(string $user, $id = null, bool $subject = false): array {
|
||||||
// if the specified ID is not a non-negative integer (or null), this will always fail
|
// if the specified ID is not a non-negative integer (or null), this will always fail
|
||||||
if (!ValueInfo::id($id, true)) {
|
if (!V::id($id, true)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "folder", 'type' => "int >= 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "folder", 'type' => "int >= 0"]);
|
||||||
}
|
}
|
||||||
// if a null or zero ID is specified this is a no-op
|
// if a null or zero ID is specified this is a no-op
|
||||||
|
@ -615,13 +626,13 @@ class Database {
|
||||||
// the root cannot be moved
|
// the root cannot be moved
|
||||||
throw new Db\ExceptionInput("circularDependence", $errData);
|
throw new Db\ExceptionInput("circularDependence", $errData);
|
||||||
}
|
}
|
||||||
$info = ValueInfo::int($parent);
|
$info = V::int($parent);
|
||||||
// the root is always a valid parent
|
// the root is always a valid parent
|
||||||
if ($info & (ValueInfo::NULL | ValueInfo::ZERO)) {
|
if ($info & (V::NULL | V::ZERO)) {
|
||||||
$parent = null;
|
$parent = null;
|
||||||
} else {
|
} else {
|
||||||
// if a negative integer or non-integer is specified this will always fail
|
// if a negative integer or non-integer is specified this will always fail
|
||||||
if (!($info & ValueInfo::VALID) || (($info & ValueInfo::NEG))) {
|
if (!($info & V::VALID) || (($info & V::NEG))) {
|
||||||
throw new Db\ExceptionInput("idMissing", $errData);
|
throw new Db\ExceptionInput("idMissing", $errData);
|
||||||
}
|
}
|
||||||
$parent = (int) $parent;
|
$parent = (int) $parent;
|
||||||
|
@ -668,12 +679,12 @@ class Database {
|
||||||
* @param integer|null $parent The parent folder context in which to check for duplication
|
* @param integer|null $parent The parent folder context in which to check for duplication
|
||||||
*/
|
*/
|
||||||
protected function folderValidateName($name, bool $checkDuplicates = false, $parent = null): bool {
|
protected function folderValidateName($name, bool $checkDuplicates = false, $parent = null): bool {
|
||||||
$info = ValueInfo::str($name);
|
$info = V::str($name);
|
||||||
if ($info & (ValueInfo::NULL | ValueInfo::EMPTY)) {
|
if ($info & (V::NULL | V::EMPTY)) {
|
||||||
throw new Db\ExceptionInput("missing", ["action" => $this->caller(), "field" => "name"]);
|
throw new Db\ExceptionInput("missing", ["action" => $this->caller(), "field" => "name"]);
|
||||||
} elseif ($info & ValueInfo::WHITE) {
|
} elseif ($info & V::WHITE) {
|
||||||
throw new Db\ExceptionInput("whitespace", ["action" => $this->caller(), "field" => "name"]);
|
throw new Db\ExceptionInput("whitespace", ["action" => $this->caller(), "field" => "name"]);
|
||||||
} elseif (!($info & ValueInfo::VALID)) {
|
} elseif (!($info & V::VALID)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "name", 'type' => "string"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "name", 'type' => "string"]);
|
||||||
} elseif ($checkDuplicates) {
|
} elseif ($checkDuplicates) {
|
||||||
// make sure that a folder with the same prospective name and parent does not already exist: if the parent is null,
|
// make sure that a folder with the same prospective name and parent does not already exist: if the parent is null,
|
||||||
|
@ -778,7 +789,7 @@ class Database {
|
||||||
* configurable retention period for newsfeeds
|
* configurable retention period for newsfeeds
|
||||||
*/
|
*/
|
||||||
public function subscriptionRemove(string $user, $id): bool {
|
public function subscriptionRemove(string $user, $id): bool {
|
||||||
if (!ValueInfo::id($id)) {
|
if (!V::id($id)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "feed", 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "feed", 'type' => "int > 0"]);
|
||||||
}
|
}
|
||||||
$changes = $this->db->prepare("DELETE from arsse_subscriptions where owner = ? and id = ?", "str", "int")->run($user, $id)->changes();
|
$changes = $this->db->prepare("DELETE from arsse_subscriptions where owner = ? and id = ?", "str", "int")->run($user, $id)->changes();
|
||||||
|
@ -807,7 +818,7 @@ class Database {
|
||||||
* - "unread": The number of unread articles associated with the subscription
|
* - "unread": The number of unread articles associated with the subscription
|
||||||
*/
|
*/
|
||||||
public function subscriptionPropertiesGet(string $user, $id): array {
|
public function subscriptionPropertiesGet(string $user, $id): array {
|
||||||
if (!ValueInfo::id($id)) {
|
if (!V::id($id)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "feed", 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "feed", 'type' => "int > 0"]);
|
||||||
}
|
}
|
||||||
$sub = $this->subscriptionList($user, null, true, (int) $id)->getRow();
|
$sub = $this->subscriptionList($user, null, true, (int) $id)->getRow();
|
||||||
|
@ -841,12 +852,12 @@ class Database {
|
||||||
if (array_key_exists("title", $data)) {
|
if (array_key_exists("title", $data)) {
|
||||||
// if the title is null, this signals intended use of the default title; otherwise make sure it's not effectively an empty string
|
// if the title is null, this signals intended use of the default title; otherwise make sure it's not effectively an empty string
|
||||||
if (!is_null($data['title'])) {
|
if (!is_null($data['title'])) {
|
||||||
$info = ValueInfo::str($data['title']);
|
$info = V::str($data['title']);
|
||||||
if ($info & ValueInfo::EMPTY) {
|
if ($info & V::EMPTY) {
|
||||||
throw new Db\ExceptionInput("missing", ["action" => __FUNCTION__, "field" => "title"]);
|
throw new Db\ExceptionInput("missing", ["action" => __FUNCTION__, "field" => "title"]);
|
||||||
} elseif ($info & ValueInfo::WHITE) {
|
} elseif ($info & V::WHITE) {
|
||||||
throw new Db\ExceptionInput("whitespace", ["action" => __FUNCTION__, "field" => "title"]);
|
throw new Db\ExceptionInput("whitespace", ["action" => __FUNCTION__, "field" => "title"]);
|
||||||
} elseif (!($info & ValueInfo::VALID)) {
|
} elseif (!($info & V::VALID)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "title", 'type' => "string"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "title", 'type' => "string"]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -918,7 +929,7 @@ class Database {
|
||||||
if (!$out && $id) {
|
if (!$out && $id) {
|
||||||
throw new Db\ExceptionInput("subjectMissing", ["action" => __FUNCTION__, "field" => "feed", 'id' => $id]);
|
throw new Db\ExceptionInput("subjectMissing", ["action" => __FUNCTION__, "field" => "feed", 'id' => $id]);
|
||||||
}
|
}
|
||||||
return ValueInfo::normalize($out, ValueInfo::T_DATE | ValueInfo::M_NULL, "sql");
|
return V::normalize($out, V::T_DATE | V::M_NULL, "sql");
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Ensures the specified subscription exists and raises an exception otherwise
|
/** Ensures the specified subscription exists and raises an exception otherwise
|
||||||
|
@ -930,7 +941,7 @@ class Database {
|
||||||
* @param boolean $subject Whether the subscription is the subject (true) rather than the object (false) of the operation being performed; this only affects the semantics of the error message if validation fails
|
* @param boolean $subject Whether the subscription is the subject (true) rather than the object (false) of the operation being performed; this only affects the semantics of the error message if validation fails
|
||||||
*/
|
*/
|
||||||
protected function subscriptionValidateId(string $user, $id, bool $subject = false): array {
|
protected function subscriptionValidateId(string $user, $id, bool $subject = false): array {
|
||||||
if (!ValueInfo::id($id)) {
|
if (!V::id($id)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "feed", 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "feed", 'type' => "int > 0"]);
|
||||||
}
|
}
|
||||||
$out = $this->db->prepare("SELECT id,feed from arsse_subscriptions where id = ? and owner = ?", "int", "str")->run($id, $user)->getRow();
|
$out = $this->db->prepare("SELECT id,feed from arsse_subscriptions where id = ? and owner = ?", "int", "str")->run($id, $user)->getRow();
|
||||||
|
@ -988,7 +999,7 @@ class Database {
|
||||||
*/
|
*/
|
||||||
public function feedUpdate($feedID, bool $throwError = false): bool {
|
public function feedUpdate($feedID, bool $throwError = false): bool {
|
||||||
// check to make sure the feed exists
|
// check to make sure the feed exists
|
||||||
if (!ValueInfo::id($feedID)) {
|
if (!V::id($feedID)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "feed", 'id' => $feedID, 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => __FUNCTION__, "field" => "feed", 'id' => $feedID, 'type' => "int > 0"]);
|
||||||
}
|
}
|
||||||
$f = $this->db->prepare("SELECT url, username, password, modified, etag, err_count, scrape FROM arsse_feeds where id = ?", "int")->run($feedID)->getRow();
|
$f = $this->db->prepare("SELECT url, username, password, modified, etag, err_count, scrape FROM arsse_feeds where id = ?", "int")->run($feedID)->getRow();
|
||||||
|
@ -1328,7 +1339,7 @@ class Database {
|
||||||
} else {
|
} else {
|
||||||
// normalize requested output and sorting columns
|
// normalize requested output and sorting columns
|
||||||
$norm = function($v) {
|
$norm = function($v) {
|
||||||
return trim(strtolower(ValueInfo::normalize($v, ValueInfo::T_STRING)));
|
return trim(strtolower(V::normalize($v, V::T_STRING)));
|
||||||
};
|
};
|
||||||
$cols = array_map($norm, $cols);
|
$cols = array_map($norm, $cols);
|
||||||
// make an output column list
|
// make an output column list
|
||||||
|
@ -1798,7 +1809,7 @@ class Database {
|
||||||
* @param integer $id The identifier of the article to validate
|
* @param integer $id The identifier of the article to validate
|
||||||
*/
|
*/
|
||||||
protected function articleValidateId(string $user, $id): array {
|
protected function articleValidateId(string $user, $id): array {
|
||||||
if (!ValueInfo::id($id)) {
|
if (!V::id($id)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "article", 'type' => "int > 0"]); // @codeCoverageIgnore
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "article", 'type' => "int > 0"]); // @codeCoverageIgnore
|
||||||
}
|
}
|
||||||
$out = $this->db->prepare(
|
$out = $this->db->prepare(
|
||||||
|
@ -1825,7 +1836,7 @@ class Database {
|
||||||
* @param integer $id The identifier of the edition to validate
|
* @param integer $id The identifier of the edition to validate
|
||||||
*/
|
*/
|
||||||
protected function articleValidateEdition(string $user, int $id): array {
|
protected function articleValidateEdition(string $user, int $id): array {
|
||||||
if (!ValueInfo::id($id)) {
|
if (!V::id($id)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "edition", 'type' => "int > 0"]); // @codeCoverageIgnore
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "edition", 'type' => "int > 0"]); // @codeCoverageIgnore
|
||||||
}
|
}
|
||||||
$out = $this->db->prepare(
|
$out = $this->db->prepare(
|
||||||
|
@ -2109,10 +2120,10 @@ class Database {
|
||||||
* @param boolean $subject Whether the label is the subject (true) rather than the object (false) of the operation being performed; this only affects the semantics of the error message if validation fails
|
* @param boolean $subject Whether the label is the subject (true) rather than the object (false) of the operation being performed; this only affects the semantics of the error message if validation fails
|
||||||
*/
|
*/
|
||||||
protected function labelValidateId(string $user, $id, bool $byName, bool $checkDb = true, bool $subject = false): array {
|
protected function labelValidateId(string $user, $id, bool $byName, bool $checkDb = true, bool $subject = false): array {
|
||||||
if (!$byName && !ValueInfo::id($id)) {
|
if (!$byName && !V::id($id)) {
|
||||||
// if we're not referring to a label by name and the ID is invalid, throw an exception
|
// if we're not referring to a label by name and the ID is invalid, throw an exception
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "label", 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "label", 'type' => "int > 0"]);
|
||||||
} elseif ($byName && !(ValueInfo::str($id) & ValueInfo::VALID)) {
|
} elseif ($byName && !(V::str($id) & V::VALID)) {
|
||||||
// otherwise if we are referring to a label by name but the ID is not a string, also throw an exception
|
// otherwise if we are referring to a label by name but the ID is not a string, also throw an exception
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "label", 'type' => "string"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "label", 'type' => "string"]);
|
||||||
} elseif ($checkDb) {
|
} elseif ($checkDb) {
|
||||||
|
@ -2133,12 +2144,12 @@ class Database {
|
||||||
|
|
||||||
/** Ensures a prospective label name is syntactically valid and raises an exception otherwise */
|
/** Ensures a prospective label name is syntactically valid and raises an exception otherwise */
|
||||||
protected function labelValidateName($name): bool {
|
protected function labelValidateName($name): bool {
|
||||||
$info = ValueInfo::str($name);
|
$info = V::str($name);
|
||||||
if ($info & (ValueInfo::NULL | ValueInfo::EMPTY)) {
|
if ($info & (V::NULL | V::EMPTY)) {
|
||||||
throw new Db\ExceptionInput("missing", ["action" => $this->caller(), "field" => "name"]);
|
throw new Db\ExceptionInput("missing", ["action" => $this->caller(), "field" => "name"]);
|
||||||
} elseif ($info & ValueInfo::WHITE) {
|
} elseif ($info & V::WHITE) {
|
||||||
throw new Db\ExceptionInput("whitespace", ["action" => $this->caller(), "field" => "name"]);
|
throw new Db\ExceptionInput("whitespace", ["action" => $this->caller(), "field" => "name"]);
|
||||||
} elseif (!($info & ValueInfo::VALID)) {
|
} elseif (!($info & V::VALID)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "name", 'type' => "string"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "name", 'type' => "string"]);
|
||||||
} else {
|
} else {
|
||||||
return true;
|
return true;
|
||||||
|
@ -2381,10 +2392,10 @@ class Database {
|
||||||
* @param boolean $subject Whether the tag is the subject (true) rather than the object (false) of the operation being performed; this only affects the semantics of the error message if validation fails
|
* @param boolean $subject Whether the tag is the subject (true) rather than the object (false) of the operation being performed; this only affects the semantics of the error message if validation fails
|
||||||
*/
|
*/
|
||||||
protected function tagValidateId(string $user, $id, bool $byName, bool $checkDb = true, bool $subject = false): array {
|
protected function tagValidateId(string $user, $id, bool $byName, bool $checkDb = true, bool $subject = false): array {
|
||||||
if (!$byName && !ValueInfo::id($id)) {
|
if (!$byName && !V::id($id)) {
|
||||||
// if we're not referring to a tag by name and the ID is invalid, throw an exception
|
// if we're not referring to a tag by name and the ID is invalid, throw an exception
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "tag", 'type' => "int > 0"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "tag", 'type' => "int > 0"]);
|
||||||
} elseif ($byName && !(ValueInfo::str($id) & ValueInfo::VALID)) {
|
} elseif ($byName && !(V::str($id) & V::VALID)) {
|
||||||
// otherwise if we are referring to a tag by name but the ID is not a string, also throw an exception
|
// otherwise if we are referring to a tag by name but the ID is not a string, also throw an exception
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "tag", 'type' => "string"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "tag", 'type' => "string"]);
|
||||||
} elseif ($checkDb) {
|
} elseif ($checkDb) {
|
||||||
|
@ -2405,12 +2416,12 @@ class Database {
|
||||||
|
|
||||||
/** Ensures a prospective tag name is syntactically valid and raises an exception otherwise */
|
/** Ensures a prospective tag name is syntactically valid and raises an exception otherwise */
|
||||||
protected function tagValidateName($name): bool {
|
protected function tagValidateName($name): bool {
|
||||||
$info = ValueInfo::str($name);
|
$info = V::str($name);
|
||||||
if ($info & (ValueInfo::NULL | ValueInfo::EMPTY)) {
|
if ($info & (V::NULL | V::EMPTY)) {
|
||||||
throw new Db\ExceptionInput("missing", ["action" => $this->caller(), "field" => "name"]);
|
throw new Db\ExceptionInput("missing", ["action" => $this->caller(), "field" => "name"]);
|
||||||
} elseif ($info & ValueInfo::WHITE) {
|
} elseif ($info & V::WHITE) {
|
||||||
throw new Db\ExceptionInput("whitespace", ["action" => $this->caller(), "field" => "name"]);
|
throw new Db\ExceptionInput("whitespace", ["action" => $this->caller(), "field" => "name"]);
|
||||||
} elseif (!($info & ValueInfo::VALID)) {
|
} elseif (!($info & V::VALID)) {
|
||||||
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "name", 'type' => "string"]);
|
throw new Db\ExceptionInput("typeViolation", ["action" => $this->caller(), "field" => "name", 'type' => "string"]);
|
||||||
} else {
|
} else {
|
||||||
return true;
|
return true;
|
||||||
|
|
56
lib/User.php
56
lib/User.php
|
@ -14,6 +14,18 @@ class User {
|
||||||
public const DRIVER_NAMES = [
|
public const DRIVER_NAMES = [
|
||||||
'internal' => \JKingWeb\Arsse\User\Internal\Driver::class,
|
'internal' => \JKingWeb\Arsse\User\Internal\Driver::class,
|
||||||
];
|
];
|
||||||
|
public const PROPERTIES = [
|
||||||
|
'admin' => V::T_BOOL,
|
||||||
|
'lang' => V::T_STRING,
|
||||||
|
'tz' => V::T_STRING,
|
||||||
|
'sort_asc' => V::T_BOOL,
|
||||||
|
'theme' => V::T_STRING,
|
||||||
|
'page_size' => V::T_INT, // greater than zero
|
||||||
|
'shortcuts' => V::T_BOOL,
|
||||||
|
'gestures' => V::T_BOOL,
|
||||||
|
'stylesheet' => V::T_STRING,
|
||||||
|
'reading_time' => V::T_BOOL,
|
||||||
|
];
|
||||||
|
|
||||||
public $id = null;
|
public $id = null;
|
||||||
|
|
||||||
|
@ -115,48 +127,42 @@ class User {
|
||||||
return (new PassGen)->length(Arsse::$conf->userTempPasswordLength)->get();
|
return (new PassGen)->length(Arsse::$conf->userTempPasswordLength)->get();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function propertiesGet(string $user): array {
|
public function propertiesGet(string $user, bool $includeLarge = true): array {
|
||||||
$extra = $this->u->userPropertiesGet($user);
|
$extra = $this->u->userPropertiesGet($user, $includeLarge);
|
||||||
// synchronize the internal database
|
// synchronize the internal database
|
||||||
if (!Arsse::$db->userExists($user)) {
|
if (!Arsse::$db->userExists($user)) {
|
||||||
Arsse::$db->userAdd($user, null);
|
Arsse::$db->userAdd($user, null);
|
||||||
Arsse::$db->userPropertiesSet($user, $extra);
|
Arsse::$db->userPropertiesSet($user, $extra);
|
||||||
}
|
}
|
||||||
// retrieve from the database to get at least the user number, and anything else the driver does not provide
|
// retrieve from the database to get at least the user number, and anything else the driver does not provide
|
||||||
$out = Arsse::$db->userPropertiesGet($user);
|
$meta = Arsse::$db->userPropertiesGet($user);
|
||||||
// layer on the driver's data
|
// combine all the data
|
||||||
foreach (["tz", "admin", "sort_asc"] as $k) {
|
$out = ['num' => $meta['num']];
|
||||||
|
foreach (self::PROPERTIES as $k => $t) {
|
||||||
if (array_key_exists($k, $extra)) {
|
if (array_key_exists($k, $extra)) {
|
||||||
$out[$k] = $extra[$k] ?? $out[$k];
|
$v = $extra[$k];
|
||||||
|
} elseif (array_key_exists($k, $meta)) {
|
||||||
|
$v = $meta[$k];
|
||||||
|
} else {
|
||||||
|
$v = null;
|
||||||
}
|
}
|
||||||
}
|
$out[$k] = V::normalize($v, $t | V::M_NULL);
|
||||||
// treat language specially since it may legitimately be null
|
|
||||||
if (array_key_exists("lang", $extra)) {
|
|
||||||
$out['lang'] = $extra['lang'];
|
|
||||||
}
|
}
|
||||||
return $out;
|
return $out;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function propertiesSet(string $user, array $data): array {
|
public function propertiesSet(string $user, array $data): array {
|
||||||
$in = [];
|
$in = [];
|
||||||
if (array_key_exists("tz", $data)) {
|
foreach (self::PROPERTIES as $k => $t) {
|
||||||
if (!is_string($data['tz'])) {
|
|
||||||
throw new User\ExceptionInput("invalidTimezone", ['field' => "tz", 'value' => ""]);
|
|
||||||
} elseif (!@timezone_open($data['tz'])) {
|
|
||||||
throw new User\ExceptionInput("invalidTimezone", ['field' => "tz", 'value' => $data['tz']]);
|
|
||||||
}
|
|
||||||
$in['tz'] = $data['tz'];
|
|
||||||
}
|
|
||||||
foreach (["admin", "sort_asc"] as $k) {
|
|
||||||
if (array_key_exists($k, $data)) {
|
if (array_key_exists($k, $data)) {
|
||||||
if (($v = V::normalize($data[$k], V::T_BOOL | V::M_DROP)) === null) {
|
// TODO: handle type mistmatch exception
|
||||||
throw new User\ExceptionInput("invalidBoolean", $k);
|
$in[$k] = V::normalize($data[$k], $t | V::M_NULL | V::M_STRICT);
|
||||||
}
|
|
||||||
$in[$k] = $v;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (array_key_exists("lang", $data)) {
|
if (isset($in['tz']) && !@timezone_open($in['tz'])) {
|
||||||
$in['lang'] = V::normalize($data['lang'], V::T_STRING | V::M_NULL);
|
throw new User\ExceptionInput("invalidTimezone", ['field' => "tz", 'value' => $in['tz']]);
|
||||||
|
} elseif (isset($in['page_size']) && $in['page_size'] < 1) {
|
||||||
|
throw new User\ExceptionInput("invalidNonZeroInteger", ['field' => "page_size"]);
|
||||||
}
|
}
|
||||||
$out = $this->u->userPropertiesSet($user, $in);
|
$out = $this->u->userPropertiesSet($user, $in);
|
||||||
// synchronize the internal database
|
// synchronize the internal database
|
||||||
|
|
|
@ -65,7 +65,7 @@ interface Driver {
|
||||||
*
|
*
|
||||||
* Any other keys will be ignored.
|
* Any other keys will be ignored.
|
||||||
*/
|
*/
|
||||||
public function userPropertiesGet(string $user): array;
|
public function userPropertiesGet(string $user, bool $includeLarge = true): array;
|
||||||
|
|
||||||
/** Sets metadata about a user
|
/** Sets metadata about a user
|
||||||
*
|
*
|
||||||
|
|
|
@ -8,9 +8,6 @@ alter table arsse_tokens add column data longtext default null;
|
||||||
|
|
||||||
alter table arsse_users add column num bigint unsigned unique;
|
alter table arsse_users add column num bigint unsigned unique;
|
||||||
alter table arsse_users add column admin boolean not null default 0;
|
alter table arsse_users add column admin boolean not null default 0;
|
||||||
alter table arsse_users add column lang longtext;
|
|
||||||
alter table arsse_users add column tz varchar(44) not null default 'Etc/UTC';
|
|
||||||
alter table arsse_users add column sort_asc boolean not null default 0;
|
|
||||||
create temporary table arsse_users_existing(
|
create temporary table arsse_users_existing(
|
||||||
id text not null,
|
id text not null,
|
||||||
num serial primary key
|
num serial primary key
|
||||||
|
@ -22,6 +19,14 @@ where u.id = n.id;
|
||||||
drop table arsse_users_existing;
|
drop table arsse_users_existing;
|
||||||
alter table arsse_users modify num bigint unsigned not null;
|
alter table arsse_users modify num bigint unsigned not null;
|
||||||
|
|
||||||
|
create table arsse_user_meta(
|
||||||
|
owner varchar(255) not null,
|
||||||
|
"key" varchar(255) not null,
|
||||||
|
value longtext,
|
||||||
|
foreign key(owner) references arsse_users(id) on delete cascade on update cascade,
|
||||||
|
primary key(owner,key)
|
||||||
|
);
|
||||||
|
|
||||||
create table arsse_icons(
|
create table arsse_icons(
|
||||||
id serial primary key,
|
id serial primary key,
|
||||||
url varchar(767) unique not null,
|
url varchar(767) unique not null,
|
||||||
|
|
|
@ -8,9 +8,6 @@ alter table arsse_tokens add column data text default null;
|
||||||
|
|
||||||
alter table arsse_users add column num bigint unique;
|
alter table arsse_users add column num bigint unique;
|
||||||
alter table arsse_users add column admin smallint not null default 0;
|
alter table arsse_users add column admin smallint not null default 0;
|
||||||
alter table arsse_users add column lang text;
|
|
||||||
alter table arsse_users add column tz text not null default 'Etc/UTC';
|
|
||||||
alter table arsse_users add column sort_asc smallint not null default 0;
|
|
||||||
create temp table arsse_users_existing(
|
create temp table arsse_users_existing(
|
||||||
id text not null,
|
id text not null,
|
||||||
num bigserial
|
num bigserial
|
||||||
|
@ -23,6 +20,13 @@ where u.id = e.id;
|
||||||
drop table arsse_users_existing;
|
drop table arsse_users_existing;
|
||||||
alter table arsse_users alter column num set not null;
|
alter table arsse_users alter column num set not null;
|
||||||
|
|
||||||
|
create table arsse_user_meta(
|
||||||
|
owner text not null references arsse_users(id) on delete cascade on update cascade,
|
||||||
|
key text not null,
|
||||||
|
value text,
|
||||||
|
primary key(owner,key)
|
||||||
|
);
|
||||||
|
|
||||||
create table arsse_icons(
|
create table arsse_icons(
|
||||||
id bigserial primary key,
|
id bigserial primary key,
|
||||||
url text unique not null,
|
url text unique not null,
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
-- This is a speculative addition to support OAuth login in the future
|
-- This is a speculative addition to support OAuth login in the future
|
||||||
alter table arsse_tokens add column data text default null;
|
alter table arsse_tokens add column data text default null;
|
||||||
|
|
||||||
-- Add multiple columns to the users table
|
-- Add num and admin columns to the users table
|
||||||
-- In particular this adds a numeric identifier for each user, which Miniflux requires
|
-- In particular this adds a numeric identifier for each user, which Miniflux requires
|
||||||
create table arsse_users_new(
|
create table arsse_users_new(
|
||||||
-- users
|
-- users
|
||||||
|
@ -14,9 +14,6 @@ create table arsse_users_new(
|
||||||
password text, -- password, salted and hashed; if using external authentication this would be blank
|
password text, -- password, salted and hashed; if using external authentication this would be blank
|
||||||
num integer unique not null, -- numeric identfier used by Miniflux
|
num integer unique not null, -- numeric identfier used by Miniflux
|
||||||
admin boolean not null default 0, -- Whether the user is an administrator
|
admin boolean not null default 0, -- Whether the user is an administrator
|
||||||
lang text, -- The user's chosen language code e.g. 'en', 'fr-ca'; null uses the system default
|
|
||||||
tz text not null default 'Etc/UTC', -- The user's chosen time zone, in zoneinfo format
|
|
||||||
sort_asc boolean not null default 0 -- Whether the user prefers to sort articles in ascending order
|
|
||||||
) without rowid;
|
) without rowid;
|
||||||
create temp table arsse_users_existing(
|
create temp table arsse_users_existing(
|
||||||
id text not null,
|
id text not null,
|
||||||
|
@ -31,6 +28,17 @@ drop table arsse_users;
|
||||||
drop table arsse_users_existing;
|
drop table arsse_users_existing;
|
||||||
alter table arsse_users_new rename to arsse_users;
|
alter table arsse_users_new rename to arsse_users;
|
||||||
|
|
||||||
|
-- Add a table for other user metadata
|
||||||
|
create table arsse_user_meta(
|
||||||
|
-- Metadata for users
|
||||||
|
-- It is up to individual applications (i.e. the client protocols) to cooperate with names and types
|
||||||
|
owner text not null references arsse_users(id) on delete cascade on update cascade, -- the user to whom the metadata belongs
|
||||||
|
key text not null, -- metadata key
|
||||||
|
value text, -- metadata value
|
||||||
|
primary key(owner,key)
|
||||||
|
) without rowid;
|
||||||
|
|
||||||
|
|
||||||
-- Add a separate table for feed icons and replace their URLs in the feeds table with their IDs
|
-- Add a separate table for feed icons and replace their URLs in the feeds table with their IDs
|
||||||
create table arsse_icons(
|
create table arsse_icons(
|
||||||
-- Icons associated with feeds
|
-- Icons associated with feeds
|
||||||
|
|
|
@ -17,14 +17,26 @@ trait SeriesUser {
|
||||||
'password' => 'str',
|
'password' => 'str',
|
||||||
'num' => 'int',
|
'num' => 'int',
|
||||||
'admin' => 'bool',
|
'admin' => 'bool',
|
||||||
'lang' => 'str',
|
|
||||||
'tz' => 'str',
|
|
||||||
'sort_asc' => 'bool',
|
|
||||||
],
|
],
|
||||||
'rows' => [
|
'rows' => [
|
||||||
["admin@example.net", '$2y$10$PbcG2ZR3Z8TuPzM7aHTF8.v61dtCjzjK78gdZJcp4UePE8T9jEgBW',1, 1, "en", "America/Toronto", 0], // password is hash of "secret"
|
["admin@example.net", '$2y$10$PbcG2ZR3Z8TuPzM7aHTF8.v61dtCjzjK78gdZJcp4UePE8T9jEgBW', 1, 1], // password is hash of "secret"
|
||||||
["jane.doe@example.com", "",2, 0, "fr", "Asia/Kuala_Lumpur", 1],
|
["jane.doe@example.com", "", 2, 0],
|
||||||
["john.doe@example.com", "",3, 0, null, "Etc/UTC", 0],
|
["john.doe@example.com", "", 3, 0],
|
||||||
|
],
|
||||||
|
],
|
||||||
|
'arsse_user_meta' => [
|
||||||
|
'columns' => [
|
||||||
|
'owner' => "str",
|
||||||
|
'key' => "str",
|
||||||
|
'value' => "str",
|
||||||
|
],
|
||||||
|
'rows' => [
|
||||||
|
["admin@example.net", "lang", "en"],
|
||||||
|
["admin@example.net", "tz", "America/Toronto"],
|
||||||
|
["admin@example.net", "sort", "desc"],
|
||||||
|
["jane.doe@example.com", "lang", "fr"],
|
||||||
|
["jane.doe@example.com", "tz", "Asia/Kuala_Lumpur"],
|
||||||
|
["jane.doe@example.com", "sort", "asc"],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|
Loading…
Reference in a new issue