4 const DELETE_ORIGINAL
= 1;
7 * Fetch the FileStore object for a given storage group
9 static function get( $group ) {
12 if( isset( $wgFileStore[$group] ) ) {
13 $info = $wgFileStore[$group];
14 return new FileStore( $group,
17 intval( $info['hash'] ) );
23 private function __construct( $group, $directory, $path, $hash ) {
24 $this->mGroup
= $group;
25 $this->mDirectory
= $directory;
27 $this->mHashLevel
= $hash;
31 * Acquire a lock; use when performing write operations on a store.
32 * This is attached to your master database connection, so if you
33 * suffer an uncaught error the lock will be released when the
34 * connection is closed.
36 * @fixme Probably only works on MySQL. Abstract to the Database class?
38 static function lock() {
39 $fname = __CLASS__
. '::' . __FUNCTION__
;
41 $dbw = wfGetDB( DB_MASTER
);
42 $lockname = $dbw->addQuotes( FileStore
::lockName() );
43 $result = $dbw->query( "SELECT GET_LOCK($lockname, 5) AS lockstatus", $fname );
44 $row = $dbw->fetchObject( $result );
45 $dbw->freeResult( $result );
47 if( $row->lockstatus
== 1 ) {
50 wfDebug( "$fname failed to acquire lock\n" );
56 * Release the global file store lock.
58 static function unlock() {
59 $fname = __CLASS__
. '::' . __FUNCTION__
;
61 $dbw = wfGetDB( DB_MASTER
);
62 $lockname = $dbw->addQuotes( FileStore
::lockName() );
63 $result = $dbw->query( "SELECT RELEASE_LOCK($lockname)", $fname );
64 $row = $dbw->fetchObject( $result );
65 $dbw->freeResult( $result );
68 private static function lockName() {
69 global $wgDBname, $wgDBprefix;
70 return "MediaWiki.{$wgDBname}.{$wgDBprefix}FileStore";
74 * Copy a file into the file store from elsewhere in the filesystem.
75 * Should be protected by FileStore::lock() to avoid race conditions.
77 * @param $key storage key string
79 * DELETE_ORIGINAL - remove the source file on transaction commit.
81 * @throws FSException if copy can't be completed
82 * @return FSTransaction
84 function insert( $key, $sourcePath, $flags=0 ) {
85 $destPath = $this->filePath( $key );
86 return $this->copyFile( $sourcePath, $destPath, $flags );
90 * Copy a file from the file store to elsewhere in the filesystem.
91 * Should be protected by FileStore::lock() to avoid race conditions.
93 * @param $key storage key string
95 * DELETE_ORIGINAL - remove the source file on transaction commit.
97 * @throws FSException if copy can't be completed
98 * @return FSTransaction on success
100 function export( $key, $destPath, $flags=0 ) {
101 $sourcePath = $this->filePath( $key );
102 return $this->copyFile( $sourcePath, $destPath, $flags );
105 private function copyFile( $sourcePath, $destPath, $flags=0 ) {
106 $fname = __CLASS__
. '::' . __FUNCTION__
;
108 if( !file_exists( $sourcePath ) ) {
110 throw new FSException( "missing source file '$sourcePath'\n" );
113 $transaction = new FSTransaction();
115 if( $flags & self
::DELETE_ORIGINAL
) {
116 $transaction->addCommit( FSTransaction
::DELETE_FILE
, $sourcePath );
119 if( file_exists( $destPath ) ) {
120 // An identical file is already present; no need to copy.
122 if( !file_exists( dirname( $destPath ) ) ) {
123 wfSuppressWarnings();
124 $ok = mkdir( dirname( $destPath ), 0777, true );
128 throw new FSException(
129 "failed to create directory for '$destPath'\n" );
133 wfSuppressWarnings();
134 $ok = copy( $sourcePath, $destPath );
138 wfDebug( "$fname copied '$sourcePath' to '$destPath'\n" );
139 $transaction->addRollback( FSTransaction
::DELETE_FILE
, $destPath );
141 throw new FSException(
142 "$fname failed to copy '$sourcePath' to '$destPath'\n" );
150 * Delete a file from the file store.
151 * Caller's responsibility to make sure it's not being used by another row.
153 * File is not actually removed until transaction commit.
154 * Should be protected by FileStore::lock() to avoid race conditions.
156 * @param $key storage key string
157 * @throws FSException if file can't be deleted
158 * @return FSTransaction
160 function delete( $key ) {
161 $destPath = $this->filePath( $key );
162 if( false === $destPath ) {
163 throw new FSExcepton( "file store does not contain file '$key'" );
165 return FileStore
::deleteFile( $destPath );
170 * Delete a non-managed file on a transactional basis.
172 * File is not actually removed until transaction commit.
173 * Should be protected by FileStore::lock() to avoid race conditions.
175 * @param $path file to remove
176 * @throws FSException if file can't be deleted
177 * @return FSTransaction
179 * @fixme Might be worth preliminary permissions check
181 static function deleteFile( $path ) {
182 if( file_exists( $path ) ) {
183 $transaction = new FSTransaction();
184 $transaction->addCommit( FSTransaction
::DELETE_FILE
, $path );
187 throw new FSException( "cannot delete missing file '$path'" );
192 * Stream a contained file directly to HTTP output.
193 * Will throw a 404 if file is missing; 400 if invalid key.
194 * @return true on success, false on failure
196 function stream( $key ) {
197 $path = $this->filePath( $key );
198 if( $path === false ) {
199 wfHttpError( 400, "Bad request", "Invalid or badly-formed filename." );
203 if( file_exists( $path ) ) {
204 // Set the filename for more convenient save behavior from browsers
205 // FIXME: Is this safe?
206 header( 'Content-Disposition: inline; filename="' . $key . '"' );
208 require_once 'StreamFile.php';
209 wfStreamFile( $path );
211 return wfHttpError( 404, "Not found",
212 "The requested resource does not exist." );
217 * Confirm that the given file key is valid.
218 * Note that a valid key may refer to a file that does not exist.
220 * Key should consist of a 32-digit base-36 SHA-1 hash and
221 * an optional alphanumeric extension, all lowercase.
222 * The whole must not exceed 64 characters.
227 static function validKey( $key ) {
228 return preg_match( '/^[0-9a-z]{32}(\.[0-9a-z]{1,31})?$/', $key );
233 * Calculate file storage key from a file on disk.
234 * You must pass an extension to it, as some files may be calculated
235 * out of a temporary file etc.
237 * @param $path to file
239 * @return string or false if could not open file or bad extension
241 static function calculateKey( $path, $extension ) {
242 $fname = __CLASS__
. '::' . __FUNCTION__
;
244 wfSuppressWarnings();
245 $hash = sha1_file( $path );
247 if( $hash === false ) {
248 wfDebug( "$fname: couldn't hash file '$path'\n" );
252 $base36 = wfBaseConvert( $hash, 16, 36, 32 );
253 if( $extension == '' ) {
256 $key = $base36 . '.' . $extension;
260 if( self
::validKey( $key ) ) {
263 wfDebug( "$fname: generated bad key '$key'\n" );
269 * Return filesystem path to the given file.
270 * Note that the file may or may not exist.
271 * @return string or false if an invalid key
273 function filePath( $key ) {
274 if( self
::validKey( $key ) ) {
275 return $this->mDirectory
. DIRECTORY_SEPARATOR
.
276 $this->hashPath( $key, DIRECTORY_SEPARATOR
);
283 * Return URL path to the given file, if the store is public.
284 * @return string or false if not public
286 function urlPath( $key ) {
287 if( $this->mUrl
&& self
::validKey( $key ) ) {
288 return $this->mUrl
. '/' . $this->hashPath( $key, '/' );
294 private function hashPath( $key, $separator ) {
296 for( $i = 0; $i < $this->mHashLevel
; $i++
) {
300 return implode( $separator, $parts );
305 * Wrapper for file store transaction stuff.
307 * FileStore methods may return one of these for undoable operations;
308 * you can then call its rollback() or commit() methods to perform
309 * final cleanup if dependent database work fails or succeeds.
311 class FSTransaction
{
312 const DELETE_FILE
= 1;
315 * Combine more items into a fancier transaction
317 function add( FSTransaction
$transaction ) {
318 $this->mOnCommit
= array_merge(
319 $this->mOnCommit
, $transaction->mOnCommit
);
320 $this->mOnRollback
= array_merge(
321 $this->mOnRollback
, $transaction->mOnRollback
);
325 * Perform final actions for success.
326 * @return true if actions applied ok, false if errors
329 return $this->apply( $this->mOnCommit
);
333 * Perform final actions for failure.
334 * @return true if actions applied ok, false if errors
336 function rollback() {
337 return $this->apply( $this->mOnRollback
);
340 // --- Private and friend functions below...
342 function __construct() {
343 $this->mOnCommit
= array();
344 $this->mOnRollback
= array();
347 function addCommit( $action, $path ) {
348 $this->mOnCommit
[] = array( $action, $path );
351 function addRollback( $action, $path ) {
352 $this->mOnRollback
[] = array( $action, $path );
355 private function apply( $actions ) {
356 $fname = __CLASS__
. '::' . __FUNCTION__
;
358 foreach( $actions as $item ) {
359 list( $action, $path ) = $item;
360 if( $action == self
::DELETE_FILE
) {
361 wfSuppressWarnings();
362 $ok = unlink( $path );
365 wfDebug( "$fname: deleting file '$path'\n" );
367 wfDebug( "$fname: failed to delete file '$path'\n" );
368 $result = $result && $ok;
375 class FSException
extends MWException
{ }