2011-06-12 00:14:24 +04:00
< ? php
/**
* ownCloud
*
* @ author Michael Gapczynski
* @ copyright 2011 Michael Gapczynski GapczynskiM @ gmail . com
*
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
* License as published by the Free Software Foundation ; either
* version 3 of the License , or any later version .
*
* This library 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 AFFERO GENERAL PUBLIC LICENSE for more details .
*
* You should have received a copy of the GNU Affero General Public
* License along with this library . If not , see < http :// www . gnu . org / licenses />.
*
*/
2011-07-19 00:36:34 +04:00
OC_HOOK :: connect ( " OC_FILESYSTEM " , " post_delete " , " OC_SHARE " , " deleteItem " );
OC_HOOK :: connect ( " OC_FILESYSTEM " , " post_rename " , " OC_SHARE " , " renameItem " );
2011-06-12 00:14:24 +04:00
/**
* This class manages shared items within the database .
*/
class OC_SHARE {
/**
* TODO notify user a file is being shared with them ?
* Share an item , adds an entry into the database
* @ param string $item
* @ param user item shared with $uid_shared_with
*/
2011-07-06 20:12:29 +04:00
public function __construct ( $source , $uid_shared_with , $permissions , $public = false ) {
if ( $source && OC_FILESYSTEM :: file_exists ( $source ) && OC_FILESYSTEM :: is_readable ( $source )) {
2011-07-21 00:52:01 +04:00
$source = " / " . OC_USER :: getUser () . " /files " . $source ;
$uid_owner = OC_USER :: getUser ();
2011-06-12 00:14:24 +04:00
if ( $public ) {
// TODO create token for public file
$token = sha1 ( " $uid_owner - $item " );
} else {
2011-07-06 20:12:29 +04:00
$query = OC_DB :: prepare ( " INSERT INTO *PREFIX*sharing VALUES(?,?,?,?,?) " );
2011-07-28 23:31:52 +04:00
$target = " / " . $uid_shared_with . " /files/Share/ " . basename ( $source );
$check = OC_DB :: prepare ( " SELECT target FROM *PREFIX*sharing WHERE target = ? AND uid_shared_with = ? " );
$result = $check -> execute ( array ( $target , $uid_shared_with )) -> fetchAll ();
// Check if target already exists for the user, if it does append a number to the name
if ( count ( $result ) > 0 ) {
if ( $pos = strrpos ( $target , " . " )) {
$name = substr ( $target , 0 , $pos );
$ext = substr ( $target , $pos );
} else {
$name = $target ;
$ext = " " ;
2011-07-20 22:16:20 +04:00
}
2011-07-28 23:31:52 +04:00
$counter = 1 ;
while ( count ( $result ) > 0 ) {
$newTarget = $name . " _ " . $counter . $ext ;
$result = $check -> execute ( array ( $newTarget , $uid_shared_with )) -> fetchAll ();
$counter ++ ;
}
$target = $newTarget ;
2011-06-12 00:14:24 +04:00
}
2011-07-28 23:31:52 +04:00
$query -> execute ( array ( $uid_owner , $uid_shared_with , $source , $target , $permissions ));
2011-06-12 00:14:24 +04:00
}
}
}
2011-07-29 03:42:02 +04:00
/**
* Get the user and the user ' s groups and put them into an array
* @ return An array to be used by the IN operator in a query for uid_shared_with
*/
private static function getUserAndGroups () {
$self = OC_USER :: getUser ();
$groups = OC_GROUP :: getUserGroups ( $self );
array_unshift ( $groups , $self );
return $groups ;
}
2011-07-15 05:04:09 +04:00
/**
* Create a new entry in the database for a file inside a shared folder
*
* $oldTarget and $newTarget may be the same value . $oldTarget exists in case the file is being moved outside of the folder
*
* @ param $oldTarget The current target location
* @ param $newTarget The new target location
*/
public static function pullOutOfFolder ( $oldTarget , $newTarget ) {
$folders = self :: getParentFolders ( $oldTarget );
2011-07-16 21:06:59 +04:00
$source = $folders [ 'source' ] . substr ( $oldTarget , strlen ( $folders [ 'target' ]));
2011-07-15 05:04:09 +04:00
$item = self :: getItem ( $folders [ 'target' ]);
$query = OC_DB :: prepare ( " INSERT INTO *PREFIX*sharing VALUES(?,?,?,?,?) " );
2011-07-21 00:52:01 +04:00
$query -> execute ( array ( $item [ 0 ][ 'uid_owner' ], OC_USER :: getUser (), $source , $newTarget , $item [ 0 ][ 'is_writeable' ]));
2011-07-15 05:04:09 +04:00
}
2011-06-12 00:14:24 +04:00
/**
2011-07-15 03:24:48 +04:00
* Get the item with the specified target location
* @ param $target The target location of the item
* @ return An array with the item
*/
public static function getItem ( $target ) {
$query = OC_DB :: prepare ( " SELECT uid_owner, source, is_writeable FROM *PREFIX*sharing WHERE target = ? AND uid_shared_with = ? LIMIT 1 " );
2011-07-21 00:52:01 +04:00
return $query -> execute ( array ( $target , OC_USER :: getUser ())) -> fetchAll ();
2011-06-12 00:14:24 +04:00
}
/**
2011-07-15 03:24:48 +04:00
* Get all items the current user is sharing
* @ return An array with all items the user is sharing
2011-06-12 00:14:24 +04:00
*/
2011-07-15 03:24:48 +04:00
public static function getMySharedItems () {
$query = OC_DB :: prepare ( " SELECT uid_shared_with, source, is_writeable FROM *PREFIX*sharing WHERE uid_owner = ? " );
2011-07-21 00:52:01 +04:00
return $query -> execute ( array ( OC_USER :: getUser ())) -> fetchAll ();
2011-06-12 00:14:24 +04:00
}
/**
2011-07-15 03:24:48 +04:00
* Get the items within a shared folder that have their own entry for the purpose of name , location , or permissions that differ from the folder itself
*
2011-07-23 22:41:01 +04:00
* Works for both target and source folders . Can be used for getting all items shared with you e . g . pass '/MTGap/files'
2011-07-15 03:24:48 +04:00
*
2011-07-23 22:41:01 +04:00
* @ param $folder The folder of the items to look for
* @ return An array with all items in the database that are in the folder
2011-07-15 03:24:48 +04:00
*/
2011-07-23 22:41:01 +04:00
public static function getItemsInFolder ( $folder ) {
2011-07-15 03:24:48 +04:00
// Append '/' in order to filter out the folder itself if not already there
2011-07-23 22:41:01 +04:00
if ( substr ( $folder , - 1 ) !== " / " ) {
$folder .= " / " ;
2011-06-12 00:14:24 +04:00
}
2011-07-16 21:06:59 +04:00
// Remove any duplicate '/'
2011-07-23 22:41:01 +04:00
$folder = preg_replace ( '{(/)\1+}' , " / " , $folder );
$length = strlen ( $folder );
2011-07-29 03:42:02 +04:00
$userAndGroups = self :: getUserAndGroups ();
2011-07-29 04:05:18 +04:00
$query = OC_DB :: prepare ( " SELECT uid_owner, source, target FROM *PREFIX*sharing WHERE SUBSTR(source, 1, ?) = ? OR SUBSTR(target, 1, ?) = ? AND uid_shared_with IN( " . substr ( str_repeat ( " ,? " , count ( $userAndGroups )), 1 ) . " ) " );
2011-07-29 03:42:02 +04:00
return $query -> execute ( array_merge ( array ( $length , $folder , $length , $folder ), $userAndGroups )) -> fetchAll ();
2011-06-12 00:14:24 +04:00
}
2011-07-06 23:17:03 +04:00
/**
2011-07-15 03:24:48 +04:00
* Get the source and target parent folders of the specified target location
* @ param $target The target location of the item
* @ return An array with the keys 'source' and 'target' with the values of the source and target parent folders
2011-07-06 20:12:29 +04:00
*/
2011-07-15 03:24:48 +04:00
public static function getParentFolders ( $target ) {
// Remove any duplicate or trailing '/'
$target = rtrim ( $target , " / " );
$target = preg_replace ( '{(/)\1+}' , " / " , $target );
2011-07-29 03:42:02 +04:00
$userAndGroups = self :: getUserAndGroups ();
2011-07-29 04:05:18 +04:00
$query = OC_DB :: prepare ( " SELECT source FROM *PREFIX*sharing WHERE target = ? AND uid_shared_with IN( " . substr ( str_repeat ( " ,? " , count ( $userAndGroups )), 1 ) . " ) LIMIT 1 " );
2011-07-15 03:24:48 +04:00
// Prevent searching for user directory e.g. '/MTGap/files'
$userDirectory = substr ( $target , 0 , strpos ( $target , " files " ) + 5 );
while ( $target != " " && $target != " / " && $target != " . " && $target != $userDirectory ) {
2011-07-15 05:04:09 +04:00
// Check if the parent directory of this target location is shared
$target = dirname ( $target );
2011-07-29 03:42:02 +04:00
$result = $query -> execute ( array_merge ( array ( $target ), $userAndGroups )) -> fetchAll ();
2011-07-15 03:24:48 +04:00
if ( count ( $result ) > 0 ) {
break ;
}
}
if ( count ( $result ) > 0 ) {
// Return both the source folder and the target folder
return array ( " source " => $result [ 0 ][ 'source' ], " target " => $target );
} else {
return false ;
}
2011-07-06 20:12:29 +04:00
}
2011-07-15 03:24:48 +04:00
2011-06-16 22:40:21 +04:00
/**
2011-07-15 03:24:48 +04:00
* Get the source location of the item at the specified target location
* @ param $target The target location of the item
* @ return Source location or false if target location is not valid
2011-06-16 22:40:21 +04:00
*/
public static function getSource ( $target ) {
2011-07-10 22:41:27 +04:00
// Remove any duplicate or trailing '/'
2011-07-05 19:56:02 +04:00
$target = rtrim ( $target , " / " );
2011-07-10 22:41:27 +04:00
$target = preg_replace ( '{(/)\1+}' , " / " , $target );
2011-07-29 03:42:02 +04:00
$userAndGroups = self :: getUserAndGroups ();
2011-07-29 04:05:18 +04:00
$query = OC_DB :: prepare ( " SELECT source FROM *PREFIX*sharing WHERE target = ? AND uid_shared_with IN( " . substr ( str_repeat ( " ,? " , count ( $userAndGroups )), 1 ) . " ) LIMIT 1 " );
2011-07-29 03:42:02 +04:00
$result = $query -> execute ( array_merge ( array ( $target ), $userAndGroups )) -> fetchAll ();
2011-06-25 03:38:39 +04:00
if ( count ( $result ) > 0 ) {
2011-06-26 06:00:52 +04:00
return $result [ 0 ][ 'source' ];
2011-06-25 03:38:39 +04:00
} else {
2011-07-15 03:24:48 +04:00
$folders = self :: getParentFolders ( $target );
2011-07-13 04:18:08 +04:00
if ( $folders == false ) {
return false ;
} else {
return $folders [ 'source' ] . substr ( $target , strlen ( $folders [ 'target' ]));
}
2011-07-06 20:12:29 +04:00
}
}
2011-07-15 03:24:48 +04:00
/**
* Check if the user has write permission for the item at the specified target location
* @ param $target The target location of the item
* @ return True if the user has write permission or false if read only
*/
public static function isWriteable ( $target ) {
2011-07-29 03:42:02 +04:00
$userAndGroups = self :: getUserAndGroups ();
2011-07-29 04:05:18 +04:00
$query = OC_DB :: prepare ( " SELECT is_writeable FROM *PREFIX*sharing WHERE target = ? AND uid_shared_with IN( " . substr ( str_repeat ( " ,? " , count ( $userAndGroups )), 1 ) . " ) LIMIT 1 " );
2011-07-29 03:42:02 +04:00
$result = $query -> execute ( array_merge ( array ( $target ), $userAndGroups )) -> fetchAll ();
2011-07-15 03:24:48 +04:00
if ( count ( $result ) > 0 ) {
return $result [ 0 ][ 'is_writeable' ];
2011-07-06 20:12:29 +04:00
} else {
2011-07-15 03:24:48 +04:00
// Check if the folder is writeable
$folders = OC_SHARE :: getParentFolders ( $target );
2011-07-29 03:42:02 +04:00
$result = $query -> execute ( array_merge ( array ( $target ), $userAndGroups )) -> fetchAll ();
2011-07-06 20:12:29 +04:00
if ( count ( $result ) > 0 ) {
2011-07-15 03:24:48 +04:00
return $result [ 0 ][ 'is_writeable' ];
2011-07-06 20:12:29 +04:00
} else {
2011-07-15 03:24:48 +04:00
return false ;
2011-07-06 20:12:29 +04:00
}
}
2011-06-16 22:40:21 +04:00
}
2011-07-09 02:21:20 +04:00
/**
2011-07-15 03:24:48 +04:00
* Set the source location to a new value
* @ param $oldSource The current source location
* @ param $newTarget The new source location
2011-07-09 02:21:20 +04:00
*/
2011-07-15 03:24:48 +04:00
public static function setSource ( $oldSource , $newSource ) {
$query = OC_DB :: prepare ( " UPDATE *PREFIX*sharing SET source = REPLACE(source, ?, ?) WHERE uid_owner = ? " );
2011-07-21 00:52:01 +04:00
$query -> execute ( array ( $oldSource , $newSource , OC_USER :: getUser ()));
2011-07-09 02:21:20 +04:00
}
2011-06-16 22:40:21 +04:00
2011-06-26 06:00:52 +04:00
/**
* Set the target location to a new value
2011-07-15 03:24:48 +04:00
*
2011-07-15 05:04:09 +04:00
* You must use the pullOutOfFolder () function to change the target location of a file inside a shared folder if the target location differs from the folder
2011-07-15 03:24:48 +04:00
*
2011-06-26 06:00:52 +04:00
* @ param $oldTarget The current target location
* @ param $newTarget The new target location
*/
public static function setTarget ( $oldTarget , $newTarget ) {
2011-07-15 03:24:48 +04:00
$query = OC_DB :: prepare ( " UPDATE *PREFIX*sharing SET target = REPLACE(target, ?, ?) WHERE uid_shared_with = ? " );
2011-07-21 00:52:01 +04:00
$query -> execute ( array ( $oldTarget , $newTarget , OC_USER :: getUser ()));
2011-06-26 06:00:52 +04:00
}
2011-06-12 00:14:24 +04:00
/**
2011-07-15 03:24:48 +04:00
* Change write permission for the specified item and user
*
* You must construct a new shared item to change the write permission of a file inside a shared folder if the write permission differs from the folder
*
* @ param $source The source location of the item
* @ param $uid_shared_with Array of users to change the write permission for
* @ param $is_writeable True if the user has write permission or false if read only
*/
public static function setIsWriteable ( $source , $uid_shared_with , $is_writeable ) {
2011-07-23 04:41:16 +04:00
$query = OC_DB :: prepare ( " UPDATE *PREFIX*sharing SET is_writeable = ? WHERE SUBSTR(source, 1, ?) = ? AND uid_shared_with = ? AND uid_owner = ? " );
2011-07-28 23:31:52 +04:00
$query -> execute ( array ( $is_writeable , strlen ( $source ), $source , $uid_shared_with , OC_USER :: getUser ()));
2011-06-12 00:14:24 +04:00
}
/**
2011-07-15 03:24:48 +04:00
* Unshare the item , removes it from all specified users
*
2011-07-15 05:04:09 +04:00
* You must use the pullOutOfFolder () function to unshare a file inside a shared folder and set $newTarget to nothing
2011-07-15 03:24:48 +04:00
*
* @ param $source The source location of the item
* @ param $uid_shared_with Array of users to unshare the item from
*/
public static function unshare ( $source , $uid_shared_with ) {
2011-07-24 22:38:01 +04:00
$query = OC_DB :: prepare ( " DELETE FROM *PREFIX*sharing WHERE SUBSTR(source, 1, ?) = ? AND uid_shared_with = ? AND uid_owner = ? " );
2011-07-28 23:31:52 +04:00
$query -> execute ( array ( strlen ( $source ), $source , $uid_shared_with , OC_USER :: getUser ()));
2011-06-12 00:14:24 +04:00
}
2011-07-15 03:24:48 +04:00
/**
* Unshare the item from the current user , removes it only from the database and doesn ' t touch the source file
*
2011-07-15 05:04:09 +04:00
* You must use the pullOutOfFolder () function to unshare a file inside a shared folder and set $newTarget to nothing
2011-07-15 03:24:48 +04:00
*
* @ param $target The target location of the item
*/
public static function unshareFromMySelf ( $target ) {
2011-07-23 04:41:16 +04:00
$query = OC_DB :: prepare ( " DELETE FROM *PREFIX*sharing WHERE SUBSTR(target, 1, ?) = ? AND uid_shared_with = ? " );
$query -> execute ( array ( strlen ( $target ), $target , OC_USER :: getUser ()));
2011-07-15 03:24:48 +04:00
}
2011-07-19 00:36:34 +04:00
/**
* Remove the item from the database , the owner deleted the file
* @ param $arguments Array of arguments passed from OC_HOOK
*/
public static function deleteItem ( $arguments ) {
2011-07-21 00:52:01 +04:00
$source = " / " . OC_USER :: getUser () . " /files " . $arguments [ 'path' ];
2011-07-23 04:41:16 +04:00
$query = OC_DB :: prepare ( " DELETE FROM *PREFIX*sharing WHERE SUBSTR(source, 1, ?) = ? AND uid_owner = ? " );
$query -> execute ( array ( strlen ( $source ), $source , OC_USER :: getUser ()));
2011-07-19 00:36:34 +04:00
}
/**
* Rename the item in the database , the owner renamed the file
* @ param $arguments Array of arguments passed from OC_HOOK
*/
public static function renameItem ( $arguments ) {
2011-07-21 00:52:01 +04:00
$oldSource = " / " . OC_USER :: getUser () . " /files " . $arguments [ 'oldpath' ];
$newSource = " / " . OC_USER :: getUser () . " /files " . $arguments [ 'newpath' ];
2011-07-23 04:41:16 +04:00
self :: setSource ( $oldSource , $newSource );
2011-07-19 00:36:34 +04:00
}
2011-06-12 00:14:24 +04:00
}
?>