2016-10-02 21:07:17 +00:00
< ? php
2016-10-06 02:08:43 +00:00
declare ( strict_types = 1 );
2016-10-02 21:07:17 +00:00
namespace JKingWeb\NewsSync ;
class Database {
2016-10-15 13:45:23 +00:00
const SCHEMA_VERSION = 1 ;
2016-10-17 20:49:39 +00:00
const FORMAT_TS = " Y-m-d h:i:s " ;
const FORMAT_DATE = " Y-m-d " ;
const FORMAT_TIME = " h:i:s " ;
2016-10-15 13:45:23 +00:00
2016-10-17 20:49:39 +00:00
protected $data ;
2016-10-28 12:27:35 +00:00
public $db ;
2016-10-15 13:45:23 +00:00
2016-10-17 20:49:39 +00:00
protected function cleanName ( string $name ) : string {
2016-10-15 13:45:23 +00:00
return ( string ) preg_filter ( " [^0-9a-zA-Z_ \ .] " , " " , $name );
}
2016-10-02 21:07:17 +00:00
2016-10-17 20:49:39 +00:00
public function __construct ( RuntimeData $data ) {
$this -> data = $data ;
2016-10-28 12:27:35 +00:00
$driver = $data -> conf -> dbDriver ;
2016-10-17 20:49:39 +00:00
$this -> db = $driver :: create ( $data , INSTALL );
2016-10-15 13:45:23 +00:00
$ver = $this -> db -> schemaVersion ();
2016-10-17 20:49:39 +00:00
if ( ! INSTALL && $ver < self :: SCHEMA_VERSION ) {
$this -> db -> update ( self :: SCHEMA_VERSION );
2016-10-15 13:45:23 +00:00
}
2016-10-02 21:07:17 +00:00
}
2016-10-06 02:08:43 +00:00
static public function listDrivers () : array {
$sep = \DIRECTORY_SEPARATOR ;
$path = __DIR__ . $sep . " Db " . $sep ;
$classes = [];
foreach ( glob ( $path . " Driver?*.php " ) as $file ) {
$name = basename ( $file , " .php " );
if ( substr ( $name , - 3 ) != " PDO " ) {
$name = NS_BASE . " Db \\ $name " ;
if ( class_exists ( $name )) {
$classes [ $name ] = $name :: driverName ();
}
}
}
return $classes ;
}
public function schemaVersion () : int {
2016-10-15 13:45:23 +00:00
return $this -> db -> schemaVersion ();
}
2016-10-18 15:42:21 +00:00
public function dbUpdate () : bool {
if ( $this -> db -> schemaVersion () < self :: SCHEMA_VERSION ) return $this -> db -> update ( self :: SCHEMA_VERSION );
return false ;
}
public function settingGet ( string $key ) {
2016-10-17 20:49:39 +00:00
$row = $this -> db -> prepare ( " SELECT value, type from newssync_settings where key = ? " , " str " ) -> run ( $key ) -> get ();
if ( ! $row ) return null ;
switch ( $row [ 'type' ]) {
case " int " : return ( int ) $row [ 'value' ];
case " numeric " : return ( float ) $row [ 'value' ];
case " text " : return $row [ 'value' ];
case " json " : return json_decode ( $row [ 'value' ]);
case " timestamp " : return date_create_from_format ( " ! " . self :: FORMAT_TS , $row [ 'value' ], new DateTimeZone ( " UTC " ));
case " date " : return date_create_from_format ( " ! " . self :: FORMAT_DATE , $row [ 'value' ], new DateTimeZone ( " UTC " ));
case " time " : return date_create_from_format ( " ! " . self :: FORMAT_TIME , $row [ 'value' ], new DateTimeZone ( " UTC " ));
case " bool " : return ( bool ) $row [ 'value' ];
case " null " : return null ;
default : return $row [ 'value' ];
}
}
2016-10-18 15:42:21 +00:00
public function settingSet ( string $key , $in , string $type = null ) : bool {
2016-10-17 20:49:39 +00:00
if ( ! $type ) {
switch ( gettype ( $in )) {
case " boolean " : $type = " bool " ; break ;
case " integer " : $type = " int " ; break ;
case " double " : $type = " numeric " ; break ;
case " string " :
case " array " : $type = " json " ; break ;
case " resource " :
case " unknown type " :
case " NULL " : $type = " null " ; break ;
case " object " :
if ( $in instanceof DateTimeInterface ) {
$type = " timestamp " ;
} else {
$type = " text " ;
}
break ;
default : $type = 'null' ; break ;
}
}
$type = strtolower ( $type );
switch ( $type ) {
case " integer " :
$type = " int " ;
case " int " :
$value =& $in ;
break ;
case " float " :
case " double " :
case " real " :
$type = " numeric " ;
case " numeric " :
$value =& $in ;
break ;
case " str " :
case " string " :
$type = " text " ;
case " text " :
$value =& $in ;
break ;
case " json " :
if ( is_array ( $in ) || is_object ( $in )) {
$value = json_encode ( $in );
} else {
$value =& $in ;
}
break ;
case " datetime " :
$type = " timestamp " ;
case " timestamp " :
if ( $in instanceof DateTimeInterface ) {
$value = gmdate ( self :: FORMAT_TS , $in -> format ( " U " ));
} else if ( is_numeric ( $in )) {
$value = gmdate ( self :: FORMAT_TS , $in );
} else {
$value = gmdate ( self :: FORMAT_TS , gmstrftime ( $in ));
}
break ;
case " date " :
if ( $in instanceof DateTimeInterface ) {
$value = gmdate ( self :: FORMAT_DATE , $in -> format ( " U " ));
} else if ( is_numeric ( $in )) {
$value = gmdate ( self :: FORMAT_DATE , $in );
} else {
$value = gmdate ( self :: FORMAT_DATE , gmstrftime ( $in ));
}
break ;
case " time " :
if ( $in instanceof DateTimeInterface ) {
$value = gmdate ( self :: FORMAT_TIME , $in -> format ( " U " ));
} else if ( is_numeric ( $in )) {
$value = gmdate ( self :: FORMAT_TIME , $in );
} else {
$value = gmdate ( self :: FORMAT_TIME , gmstrftime ( $in ));
}
break ;
case " boolean " :
case " bit " :
$type = " bool " ;
case " bool " :
$value = ( int ) $in ;
break ;
case " null " :
$value = null ;
break ;
default :
$type = " text " ;
$value =& $in ;
break ;
}
$this -> db -> prepare ( " REPLACE INTO newssync_settings(key,value,type) values(?,?,?) " , " str " , (( $type == " null " ) ? " null " : " str " ), " str " ) -> run ( $key , $value , " text " );
}
2016-10-28 12:27:35 +00:00
public function settingRemove ( string $key ) : bool {
2016-10-17 20:49:39 +00:00
$this -> db -> prepare ( " DELETE from newssync_settings where key = ? " , " str " ) -> run ( $key );
return true ;
}
2016-11-04 02:54:27 +00:00
public function userExists ( string $user ) : bool {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
return ( bool ) $this -> db -> prepare ( " SELECT count(*) from newssync_users where id is ? " , " str " ) -> run ( $user ) -> getSingle ();
2016-10-15 13:45:23 +00:00
}
2016-10-18 15:42:21 +00:00
2016-11-04 02:54:27 +00:00
public function userAdd ( string $user , string $password = null ) : bool {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
if ( $this -> userExists ( $user )) return false ;
2016-10-28 12:27:35 +00:00
if ( strlen ( $password ) > 0 ) $password = password_hash ( $password , \PASSWORD_DEFAULT );
2016-11-04 02:54:27 +00:00
$this -> db -> prepare ( " INSERT INTO newssync_users(id,password) values(?,?) " , " str " , " str " , " str " ) -> run ( $user , $password , $admin );
2016-10-28 12:27:35 +00:00
return true ;
}
2016-11-04 02:54:27 +00:00
public function userRemove ( string $user ) : bool {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
$this -> db -> prepare ( " DELETE from newssync_users where id is ? " , " str " ) -> run ( $user );
2016-10-28 12:27:35 +00:00
return true ;
}
public function userList ( string $domain = null ) : array {
if ( $domain !== null ) {
2016-11-04 02:54:27 +00:00
if ( ! $this -> data -> user -> authorize ( " @ " . $domain , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $domain ]);
2016-10-28 12:27:35 +00:00
$domain = str_replace ([ " \\ " , " % " , " _ " ],[ " \\ \\ " , " \\ % " , " \\ _ " ], $domain );
$domain = " %@ " . $domain ;
$set = $this -> db -> prepare ( " SELECT id from newssync_users where id like ? " , " str " ) -> run ( $domain );
} else {
2016-11-04 02:54:27 +00:00
if ( ! $this -> data -> user -> authorize ( " " , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => " all users " ]);
$set = $this -> db -> prepare ( " SELECT id from newssync_users " ) -> run ();
2016-10-28 12:27:35 +00:00
}
$out = [];
foreach ( $set as $row ) {
$out [] = $row [ " id " ];
}
return $out ;
}
2016-11-04 02:54:27 +00:00
public function userPasswordGet ( string $user ) : string {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
if ( ! $this -> userExists ( $user )) return " " ;
return ( string ) $this -> db -> prepare ( " SELECT password from newssync_users where id is ? " , " str " ) -> run ( $user ) -> getSingle ();
}
public function userPasswordSet ( string $user , string $password = null ) : bool {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
if ( ! $this -> userExists ( $user )) return false ;
2016-10-28 12:27:35 +00:00
if ( strlen ( $password > 0 )) $password = password_hash ( $password );
2016-11-04 02:54:27 +00:00
$this -> db -> prepare ( " UPDATE newssync_users set password = ? where id is ? " , " str " , " str " ) -> run ( $password , $user );
2016-10-28 12:27:35 +00:00
return true ;
}
2016-11-04 02:54:27 +00:00
public function userPropertiesGet ( string $user ) : array {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
$prop = $this -> db -> prepare ( " SELECT name,rights from newssync_users where id is ? " , " str " ) -> run ( $user ) -> get ();
2016-10-28 12:27:35 +00:00
if ( ! $prop ) return [];
return $prop ;
}
2016-11-04 02:54:27 +00:00
public function userPropertiesSet ( string $user , array & $properties ) : array {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
2016-10-28 12:27:35 +00:00
$valid = [ // FIXME: add future properties
" name " => " str " ,
];
2016-11-04 02:54:27 +00:00
if ( ! $this -> userExists ( $user )) return [];
2016-10-28 12:27:35 +00:00
$this -> db -> begin ();
foreach ( $valid as $prop => $type ) {
if ( ! array_key_exists ( $prop , $properties )) continue ;
2016-11-04 02:54:27 +00:00
$this -> db -> prepare ( " UPDATE newssync_users set $prop = ? where id is ? " , $type , " str " ) -> run ( $properties [ $prop ], $user );
2016-10-28 12:27:35 +00:00
}
$this -> db -> commit ();
2016-11-04 02:54:27 +00:00
return $this -> userPropertiesGet ( $user );
}
public function userRightsGet ( string $user ) : int {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
return ( int ) $this -> db -> prepare ( " SELECT rights from newssync_users where id is ? " , " str " ) -> run ( $user ) -> getSingle ();
}
public function userRightsSet ( string $user , int $rights ) : bool {
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
if ( ! $this -> userExists ( $user )) return false ;
$this -> db -> prepare ( " UPDATE newssync_users set rights = ? where id is ? " , " int " , " str " ) -> run ( $rights , $user );
return true ;
2016-10-28 12:27:35 +00:00
}
public function subscriptionAdd ( string $user , string $url , string $fetchUser = " " , string $fetchPassword = " " ) : int {
2016-11-04 02:54:27 +00:00
if ( ! $this -> data -> user -> authorize ( $user , __FUNCTION__ )) throw new User\ExceptionAuthz ( " notAuthorized " , [ " action " => __FUNCTION__ , " user " => $user ]);
if ( ! $this -> userExists ( $user )) throw new User\Exception ( " doesNotExist " , [ " user " => $user , " action " => __FUNCTION__ ]);
2016-10-15 13:45:23 +00:00
$this -> db -> begin ();
2016-10-18 15:42:21 +00:00
$qFeed = $this -> db -> prepare ( " SELECT id from newssync_feeds where url is ? and username is ? and password is ? " , " str " , " str " , " str " );
2016-11-04 02:54:27 +00:00
$feed = $qFeed -> run ( $url , $fetchUser , $fetchPassword ) -> getSingle ();
if ( $feed === null ) {
2016-10-15 13:45:23 +00:00
$this -> db -> prepare ( " INSERT INTO newssync_feeds(url,username,password) values(?,?,?) " , " str " , " str " , " str " ) -> run ( $url , $fetchUser , $fetchPassword );
2016-11-04 02:54:27 +00:00
$feed = $qFeed -> run ( $url , $fetchUser , $fetchPassword ) -> getSingle ();
2016-10-15 13:45:23 +00:00
}
2016-11-04 02:54:27 +00:00
$this -> db -> prepare ( " INSERT INTO newssync_subscriptions(owner,feed) values(?,?) " , " str " , " int " ) -> run ( $user , $feed );
$sub = $this -> db -> prepare ( " SELECT id from newssync_subscriptions where owner is ? and feed is ? " , " str " , " int " ) -> run ( $user , $feed ) -> getSingle ();
2016-10-15 13:45:23 +00:00
$this -> db -> commit ();
2016-11-04 02:54:27 +00:00
return $sub ;
}
public function subscriptionRemove ( int $id ) : bool {
$this -> db -> begin ();
$feed = $this -> db -> prepare ( " SELECT feed from newssync_subscriptions where id is ? " , " int " ) -> run ( $id ) -> getSingle ();
$this -> db -> prepare ( " DELETE from newssync_subscriptions where id is ? " , " int " ) -> run ( $id );
2016-10-02 21:07:17 +00:00
}
2016-10-15 13:45:23 +00:00
2016-10-02 21:07:17 +00:00
}