Blame view

sources/apps/files_versions/lib/versions.php 18.4 KB
03e52840d   Kload   Init
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
  <?php
  /**
   * Copyright (c) 2012 Frank Karlitschek <frank@owncloud.org>
   *               2013 Bjoern Schiessle <schiessle@owncloud.com>
   * This file is licensed under the Affero General Public License version 3 or
   * later.
   * See the COPYING-README file.
   */
  
  /**
   * Versions
   *
   * A class to handle the versioning of files.
   */
  
  namespace OCA\Files_Versions;
  
  class Storage {
  
  	const DEFAULTENABLED=true;
  	const DEFAULTMAXSIZE=50; // unit: percentage; 50% of available disk space/quota
31b7f2792   Kload   Upgrade to ownclo...
22
23
24
25
  	const VERSIONS_ROOT = 'files_versions/';
  
  	// files for which we can remove the versions after the delete operation was successful
  	private static $deletedFiles = array();
03e52840d   Kload   Init
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
  
  	private static $max_versions_per_interval = array(
  		//first 10sec, one version every 2sec
  		1 => array('intervalEndsAfter' => 10,      'step' => 2),
  		//next minute, one version every 10sec
  		2 => array('intervalEndsAfter' => 60,      'step' => 10),
  		//next hour, one version every minute
  		3 => array('intervalEndsAfter' => 3600,    'step' => 60),
  		//next 24h, one version every hour
  		4 => array('intervalEndsAfter' => 86400,   'step' => 3600),
  		//next 30days, one version per day
  		5 => array('intervalEndsAfter' => 2592000, 'step' => 86400),
  		//until the end one version per week
  		6 => array('intervalEndsAfter' => -1,      'step' => 604800),
  	);
  
  	public static function getUidAndFilename($filename) {
  		$uid = \OC\Files\Filesystem::getOwner($filename);
  		\OC\Files\Filesystem::initMountPoints($uid);
  		if ( $uid != \OCP\User::getUser() ) {
  			$info = \OC\Files\Filesystem::getFileInfo($filename);
  			$ownerView = new \OC\Files\View('/'.$uid.'/files');
  			$filename = $ownerView->getPath($info['fileid']);
  		}
  		return array($uid, $filename);
  	}
  
  	/**
  	 * get current size of all versions from a given user
  	 *
6d9380f96   Cédric Dupont   Update sources OC...
56
57
  	 * @param string $user user who owns the versions
  	 * @return int versions size
03e52840d   Kload   Init
58
59
  	 */
  	private static function getVersionsSize($user) {
6d9380f96   Cédric Dupont   Update sources OC...
60
61
62
  		$view = new \OC\Files\View('/' . $user);
  		$fileInfo = $view->getFileInfo('/files_versions');
  		return isset($fileInfo['size']) ? $fileInfo['size'] : 0;
03e52840d   Kload   Init
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
  	}
  
  	/**
  	 * store a new version of a file.
  	 */
  	public static function store($filename) {
  		if(\OCP\Config::getSystemValue('files_versions', Storage::DEFAULTENABLED)=='true') {
  
  			// if the file gets streamed we need to remove the .part extension
  			// to get the right target
  			$ext = pathinfo($filename, PATHINFO_EXTENSION);
  			if ($ext === 'part') {
  				$filename = substr($filename, 0, strlen($filename)-5);
  			}
  
  			list($uid, $filename) = self::getUidAndFilename($filename);
  
  			$files_view = new \OC\Files\View('/'.$uid .'/files');
  			$users_view = new \OC\Files\View('/'.$uid);
03e52840d   Kload   Init
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
  
  			// check if filename is a directory
  			if($files_view->is_dir($filename)) {
  				return false;
  			}
  
  			// we should have a source file to work with, and the file shouldn't
  			// be empty
  			$fileExists = $files_view->file_exists($filename);
  			if (!($fileExists && $files_view->filesize($filename) > 0)) {
  				return false;
  			}
  
  			// create all parent folders
  			self::createMissingDirectories($filename, $users_view);
  
  			$versionsSize = self::getVersionsSize($uid);
03e52840d   Kload   Init
99
100
101
102
103
  
  			// assumption: we need filesize($filename) for the new version +
  			// some more free space for the modified file which might be
  			// 1.5 times as large as the current version -> 2.5
  			$neededSpace = $files_view->filesize($filename) * 2.5;
6d9380f96   Cédric Dupont   Update sources OC...
104
  			self::expire($filename, $versionsSize, $neededSpace);
03e52840d   Kload   Init
105
106
107
108
109
110
  
  			// disable proxy to prevent multiple fopen calls
  			$proxyStatus = \OC_FileProxy::$enabled;
  			\OC_FileProxy::$enabled = false;
  
  			// store a new version of a file
a293d369c   Kload   Update sources to...
111
112
113
114
  			$mtime = $users_view->filemtime('files'.$filename);
  			$users_view->copy('files'.$filename, 'files_versions'.$filename.'.v'. $mtime);
  			// call getFileInfo to enforce a file cache entry for the new version
  			$users_view->getFileInfo('files_versions'.$filename.'.v'.$mtime);
03e52840d   Kload   Init
115
116
117
  
  			// reset proxy state
  			\OC_FileProxy::$enabled = $proxyStatus;
03e52840d   Kload   Init
118
119
120
121
122
  		}
  	}
  
  
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
123
  	 * mark file as deleted so that we can remove the versions if the file is gone
31b7f2792   Kload   Upgrade to ownclo...
124
125
126
127
128
129
130
131
132
133
  	 * @param string $path
  	 */
  	public static function markDeletedFile($path) {
  		list($uid, $filename) = self::getUidAndFilename($path);
  		self::$deletedFiles[$path] = array(
  			'uid' => $uid,
  			'filename' => $filename);
  	}
  
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
  	 * delete the version from the storage and cache
  	 *
  	 * @param \OC\Files\View $view
  	 * @param string $path
  	 */
  	protected static function deleteVersion($view, $path) {
  		$view->unlink($path);
  		/**
  		 * @var \OC\Files\Storage\Storage $storage
  		 * @var string $internalPath
  		 */
  		list($storage, $internalPath) = $view->resolvePath($path);
  		$cache = $storage->getCache($internalPath);
  		$cache->remove($internalPath);
  	}
  
  	/**
03e52840d   Kload   Init
151
152
  	 * Delete versions of a file
  	 */
31b7f2792   Kload   Upgrade to ownclo...
153
  	public static function delete($path) {
03e52840d   Kload   Init
154

31b7f2792   Kload   Upgrade to ownclo...
155
156
157
158
159
  		$deletedFile = self::$deletedFiles[$path];
  		$uid = $deletedFile['uid'];
  		$filename = $deletedFile['filename'];
  
  		if (!\OC\Files\Filesystem::file_exists($path)) {
6d9380f96   Cédric Dupont   Update sources OC...
160
  			$view = new \OC\Files\View('/' . $uid . '/files_versions');
31b7f2792   Kload   Upgrade to ownclo...
161

31b7f2792   Kload   Upgrade to ownclo...
162
163
  			$versions = self::getVersions($uid, $filename);
  			if (!empty($versions)) {
31b7f2792   Kload   Upgrade to ownclo...
164
  				foreach ($versions as $v) {
6d9380f96   Cédric Dupont   Update sources OC...
165
166
167
  					\OC_Hook::emit('\OCP\Versions', 'preDelete', array('path' => $path . $v['version']));
  					self::deleteVersion($view, $filename . '.v' . $v['version']);
  					\OC_Hook::emit('\OCP\Versions', 'delete', array('path' => $path . $v['version']));
31b7f2792   Kload   Upgrade to ownclo...
168
  				}
03e52840d   Kload   Init
169
  			}
03e52840d   Kload   Init
170
  		}
31b7f2792   Kload   Upgrade to ownclo...
171
  		unset(self::$deletedFiles[$path]);
03e52840d   Kload   Init
172
173
174
  	}
  
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
175
176
177
178
  	 * rename or copy versions of a file
  	 * @param string $old_path
  	 * @param string $new_path
  	 * @param string $operation can be 'copy' or 'rename'
03e52840d   Kload   Init
179
  	 */
6d9380f96   Cédric Dupont   Update sources OC...
180
  	public static function renameOrCopy($old_path, $new_path, $operation) {
03e52840d   Kload   Init
181
182
183
184
185
186
187
188
189
190
  		list($uid, $oldpath) = self::getUidAndFilename($old_path);
  		list($uidn, $newpath) = self::getUidAndFilename($new_path);
  		$versions_view = new \OC\Files\View('/'.$uid .'/files_versions');
  		$files_view = new \OC\Files\View('/'.$uid .'/files');
  
  		// if the file already exists than it was a upload of a existing file
  		// over the web interface -> store() is the right function we need here
  		if ($files_view->file_exists($newpath)) {
  			return self::store($new_path);
  		}
03e52840d   Kload   Init
191
  		if ( $files_view->is_dir($oldpath) && $versions_view->is_dir($oldpath) ) {
6d9380f96   Cédric Dupont   Update sources OC...
192
  			$versions_view->$operation($oldpath, $newpath);
03e52840d   Kload   Init
193
194
195
196
197
  		} else  if ( ($versions = Storage::getVersions($uid, $oldpath)) ) {
  			// create missing dirs if necessary
  			self::createMissingDirectories($newpath, new \OC\Files\View('/'. $uidn));
  
  			foreach ($versions as $v) {
6d9380f96   Cédric Dupont   Update sources OC...
198
  				$versions_view->$operation($oldpath.'.v'.$v['version'], $newpath.'.v'.$v['version']);
03e52840d   Kload   Init
199
200
  			}
  		}
6d9380f96   Cédric Dupont   Update sources OC...
201
202
203
204
  
  		if (!$files_view->is_dir($newpath)) {
  			self::expire($newpath);
  		}
03e52840d   Kload   Init
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
  	}
  
  	/**
  	 * rollback to an old version of a file.
  	 */
  	public static function rollback($file, $revision) {
  
  		if(\OCP\Config::getSystemValue('files_versions', Storage::DEFAULTENABLED)=='true') {
  			list($uid, $filename) = self::getUidAndFilename($file);
  			$users_view = new \OC\Files\View('/'.$uid);
  			$files_view = new \OC\Files\View('/'.\OCP\User::getUser().'/files');
  			$versionCreated = false;
  
  			//first create a new version
  			$version = 'files_versions'.$filename.'.v'.$users_view->filemtime('files'.$filename);
  			if ( !$users_view->file_exists($version)) {
  
  				// disable proxy to prevent multiple fopen calls
  				$proxyStatus = \OC_FileProxy::$enabled;
  				\OC_FileProxy::$enabled = false;
  
  				$users_view->copy('files'.$filename, 'files_versions'.$filename.'.v'.$users_view->filemtime('files'.$filename));
  
  				// reset proxy state
  				\OC_FileProxy::$enabled = $proxyStatus;
  
  				$versionCreated = true;
  			}
  
  			// rollback
  			if( @$users_view->rename('files_versions'.$filename.'.v'.$revision, 'files'.$filename) ) {
  				$files_view->touch($file, $revision);
  				Storage::expire($file);
  				return true;
  
  			}else if ( $versionCreated ) {
6d9380f96   Cédric Dupont   Update sources OC...
241
  				self::deleteVersion($users_view, $version);
03e52840d   Kload   Init
242
243
244
245
246
247
248
249
  			}
  		}
  		return false;
  
  	}
  
  
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
250
  	 * get a list of all available versions of a file in descending chronological order
a293d369c   Kload   Update sources to...
251
252
253
  	 * @param string $uid user id from the owner of the file
  	 * @param string $filename file to find versions of, relative to the user files dir
  	 * @param string $userFullPath
6d9380f96   Cédric Dupont   Update sources OC...
254
  	 * @return array versions newest version first
03e52840d   Kload   Init
255
  	 */
a293d369c   Kload   Update sources to...
256
  	public static function getVersions($uid, $filename, $userFullPath = '') {
31b7f2792   Kload   Upgrade to ownclo...
257
258
  		$versions = array();
  		// fetch for old versions
6d9380f96   Cédric Dupont   Update sources OC...
259
  		$view = new \OC\Files\View('/' . $uid . '/');
31b7f2792   Kload   Upgrade to ownclo...
260
261
  
  		$pathinfo = pathinfo($filename);
6d9380f96   Cédric Dupont   Update sources OC...
262
  		$versionedFile = $pathinfo['basename'];
31b7f2792   Kload   Upgrade to ownclo...
263

f7d878ff1   kload   [enh] Update to 7...
264
  		$dir = \OC\Files\Filesystem::normalizePath(self::VERSIONS_ROOT . '/' . $pathinfo['dirname']);
31b7f2792   Kload   Upgrade to ownclo...
265

6d9380f96   Cédric Dupont   Update sources OC...
266
267
268
269
  		$dirContent = false;
  		if ($view->is_dir($dir)) {
  			$dirContent = $view->opendir($dir);
  		}
31b7f2792   Kload   Upgrade to ownclo...
270

6d9380f96   Cédric Dupont   Update sources OC...
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
  		if ($dirContent === false) {
  			return $versions;
  		}
  
  		if (is_resource($dirContent)) {
  			while (($entryName = readdir($dirContent)) !== false) {
  				if (!\OC\Files\Filesystem::isIgnoredDir($entryName)) {
  					$pathparts = pathinfo($entryName);
  					$filename = $pathparts['filename'];
  					if ($filename === $versionedFile) {
  						$pathparts = pathinfo($entryName);
  						$timestamp = substr($pathparts['extension'], 1);
  						$filename = $pathparts['filename'];
  						$key = $timestamp . '#' . $filename;
  						$versions[$key]['version'] = $timestamp;
  						$versions[$key]['humanReadableTimestamp'] = self::getHumanReadableTimestamp($timestamp);
  						if (empty($userFullPath)) {
  							$versions[$key]['preview'] = '';
  						} else {
  							$versions[$key]['preview'] = \OCP\Util::linkToRoute('core_ajax_versions_preview', array('file' => $userFullPath, 'version' => $timestamp));
  						}
f7d878ff1   kload   [enh] Update to 7...
292
  						$versions[$key]['path'] = $pathinfo['dirname'] . '/' . $filename;
6d9380f96   Cédric Dupont   Update sources OC...
293
294
  						$versions[$key]['name'] = $versionedFile;
  						$versions[$key]['size'] = $view->filesize($dir . '/' . $entryName);
a293d369c   Kload   Update sources to...
295
  					}
03e52840d   Kload   Init
296
297
  				}
  			}
6d9380f96   Cédric Dupont   Update sources OC...
298
  			closedir($dirContent);
31b7f2792   Kload   Upgrade to ownclo...
299
  		}
03e52840d   Kload   Init
300

31b7f2792   Kload   Upgrade to ownclo...
301
302
  		// sort with newest version first
  		krsort($versions);
03e52840d   Kload   Init
303

31b7f2792   Kload   Upgrade to ownclo...
304
305
  		return $versions;
  	}
03e52840d   Kload   Init
306

31b7f2792   Kload   Upgrade to ownclo...
307
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
308
  	 * translate a timestamp into a string like "5 days ago"
31b7f2792   Kload   Upgrade to ownclo...
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
  	 * @param int $timestamp
  	 * @return string for example "5 days ago"
  	 */
  	private static function getHumanReadableTimestamp($timestamp) {
  
  		$diff = time() - $timestamp;
  
  		if ($diff < 60) { // first minute
  			return  $diff . " seconds ago";
  		} elseif ($diff < 3600) { //first hour
  			return round($diff / 60) . " minutes ago";
  		} elseif ($diff < 86400) { // first day
  			return round($diff / 3600) . " hours ago";
  		} elseif ($diff < 604800) { //first week
  			return round($diff / 86400) . " days ago";
  		} elseif ($diff < 2419200) { //first month
  			return round($diff / 604800) . " weeks ago";
  		} elseif ($diff < 29030400) { // first year
  			return round($diff / 2419200) . " months ago";
03e52840d   Kload   Init
328
  		} else {
31b7f2792   Kload   Upgrade to ownclo...
329
  			return round($diff / 29030400) . " years ago";
03e52840d   Kload   Init
330
331
332
  		}
  
  	}
03e52840d   Kload   Init
333
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
334
335
  	 * returns all stored file versions from a given user
  	 * @param string $uid id of the user
03e52840d   Kload   Init
336
337
338
  	 * @return array with contains two arrays 'all' which contains all versions sorted by age and 'by_file' which contains all versions sorted by filename
  	 */
  	private static function getAllVersions($uid) {
31b7f2792   Kload   Upgrade to ownclo...
339
340
  		$view = new \OC\Files\View('/' . $uid . '/');
  		$dirs = array(self::VERSIONS_ROOT);
6d9380f96   Cédric Dupont   Update sources OC...
341
  		$versions = array();
03e52840d   Kload   Init
342

31b7f2792   Kload   Upgrade to ownclo...
343
344
345
  		while (!empty($dirs)) {
  			$dir = array_pop($dirs);
  			$files = $view->getDirectoryContent($dir);
03e52840d   Kload   Init
346

31b7f2792   Kload   Upgrade to ownclo...
347
348
349
350
351
352
353
354
355
356
  			foreach ($files as $file) {
  				if ($file['type'] === 'dir') {
  					array_push($dirs, $file['path']);
  				} else {
  					$versionsBegin = strrpos($file['path'], '.v');
  					$relPathStart = strlen(self::VERSIONS_ROOT);
  					$version = substr($file['path'], $versionsBegin + 2);
  					$relpath = substr($file['path'], $relPathStart, $versionsBegin - $relPathStart);
  					$key = $version . '#' . $relpath;
  					$versions[$key] = array('path' => $relpath, 'timestamp' => $version);
03e52840d   Kload   Init
357
358
  				}
  			}
31b7f2792   Kload   Upgrade to ownclo...
359
  		}
03e52840d   Kload   Init
360

6d9380f96   Cédric Dupont   Update sources OC...
361
  		// newest version first
a293d369c   Kload   Update sources to...
362
  		krsort($versions);
03e52840d   Kload   Init
363

31b7f2792   Kload   Upgrade to ownclo...
364
  		$result = array();
03e52840d   Kload   Init
365

31b7f2792   Kload   Upgrade to ownclo...
366
  		foreach ($versions as $key => $value) {
a293d369c   Kload   Update sources to...
367
  			$size = $view->filesize(self::VERSIONS_ROOT.'/'.$value['path'].'.v'.$value['timestamp']);
31b7f2792   Kload   Upgrade to ownclo...
368
  			$filename = $value['path'];
03e52840d   Kload   Init
369

31b7f2792   Kload   Upgrade to ownclo...
370
371
372
  			$result['all'][$key]['version'] = $value['timestamp'];
  			$result['all'][$key]['path'] = $filename;
  			$result['all'][$key]['size'] = $size;
03e52840d   Kload   Init
373

31b7f2792   Kload   Upgrade to ownclo...
374
375
376
  			$result['by_file'][$filename][$key]['version'] = $value['timestamp'];
  			$result['by_file'][$filename][$key]['path'] = $filename;
  			$result['by_file'][$filename][$key]['size'] = $size;
03e52840d   Kload   Init
377
  		}
31b7f2792   Kload   Upgrade to ownclo...
378
379
  
  		return $result;
03e52840d   Kload   Init
380
381
382
  	}
  
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
  	 * get list of files we want to expire
  	 * @param array $versions list of versions
  	 * @param integer $time
  	 * @return array containing the list of to deleted versions and the size of them
  	 */
  	protected static function getExpireList($time, $versions) {
  
  		$size = 0;
  		$toDelete = array();  // versions we want to delete
  
  		$interval = 1;
  		$step = Storage::$max_versions_per_interval[$interval]['step'];
  		if (Storage::$max_versions_per_interval[$interval]['intervalEndsAfter'] == -1) {
  			$nextInterval = -1;
  		} else {
  			$nextInterval = $time - Storage::$max_versions_per_interval[$interval]['intervalEndsAfter'];
  		}
  
  		$firstVersion = reset($versions);
  		$firstKey = key($versions);
  		$prevTimestamp = $firstVersion['version'];
  		$nextVersion = $firstVersion['version'] - $step;
  		unset($versions[$firstKey]);
  
  		foreach ($versions as $key => $version) {
  			$newInterval = true;
  			while ($newInterval) {
  				if ($nextInterval == -1 || $prevTimestamp > $nextInterval) {
  					if ($version['version'] > $nextVersion) {
  						//distance between two version too small, mark to delete
  						$toDelete[$key] = $version['path'] . '.v' . $version['version'];
  						$size += $version['size'];
  						\OCP\Util::writeLog('files_versions', 'Mark to expire '. $version['path'] .' next version should be ' . $nextVersion . " or smaller. (prevTimestamp: " . $prevTimestamp . "; step: " . $step, \OCP\Util::DEBUG);
  					} else {
  						$nextVersion = $version['version'] - $step;
  						$prevTimestamp = $version['version'];
  					}
  					$newInterval = false; // version checked so we can move to the next one
  				} else { // time to move on to the next interval
  					$interval++;
  					$step = Storage::$max_versions_per_interval[$interval]['step'];
  					$nextVersion = $prevTimestamp - $step;
  					if (Storage::$max_versions_per_interval[$interval]['intervalEndsAfter'] == -1) {
  						$nextInterval = -1;
  					} else {
  						$nextInterval = $time - Storage::$max_versions_per_interval[$interval]['intervalEndsAfter'];
  					}
  					$newInterval = true; // we changed the interval -> check same version with new interval
  				}
  			}
  		}
  
  		return array($toDelete, $size);
  
  	}
  
  	/**
  	 * Erase a file's versions which exceed the set quota
03e52840d   Kload   Init
441
442
443
444
445
446
447
448
449
450
  	 */
  	private static function expire($filename, $versionsSize = null, $offset = 0) {
  		if(\OCP\Config::getSystemValue('files_versions', Storage::DEFAULTENABLED)=='true') {
  			list($uid, $filename) = self::getUidAndFilename($filename);
  			$versionsFileview = new \OC\Files\View('/'.$uid.'/files_versions');
  
  			// get available disk space for user
  			$softQuota = true;
  			$quota = \OC_Preferences::getValue($uid, 'files', 'quota');
  			if ( $quota === null || $quota === 'default') {
6d9380f96   Cédric Dupont   Update sources OC...
451
  				$quota = \OC::$server->getAppConfig()->getValue('files', 'default_quota');
03e52840d   Kload   Init
452
453
454
455
456
457
458
459
460
461
462
  			}
  			if ( $quota === null || $quota === 'none' ) {
  				$quota = \OC\Files\Filesystem::free_space('/');
  				$softQuota = false;
  			} else {
  				$quota = \OCP\Util::computerFileSize($quota);
  			}
  
  			// make sure that we have the current size of the version history
  			if ( $versionsSize === null ) {
  				$versionsSize = self::getVersionsSize($uid);
03e52840d   Kload   Init
463
464
465
466
467
468
  			}
  
  			// calculate available space for version history
  			// subtract size of files and current versions size from quota
  			if ($softQuota) {
  				$files_view = new \OC\Files\View('/'.$uid.'/files');
a293d369c   Kload   Update sources to...
469
  				$rootInfo = $files_view->getFileInfo('/', false);
03e52840d   Kload   Init
470
471
472
473
474
475
476
477
478
  				$free = $quota-$rootInfo['size']; // remaining free space for user
  				if ( $free > 0 ) {
  					$availableSpace = ($free * self::DEFAULTMAXSIZE / 100) - ($versionsSize + $offset); // how much space can be used for versions
  				} else {
  					$availableSpace = $free - $versionsSize - $offset;
  				}
  			} else {
  				$availableSpace = $quota - $offset;
  			}
03e52840d   Kload   Init
479
  			$allVersions = Storage::getVersions($uid, $filename);
03e52840d   Kload   Init
480

6d9380f96   Cédric Dupont   Update sources OC...
481
482
  			$time = time();
  			list($toDelete, $sizeOfDeletedVersions) = self::getExpireList($time, $allVersions);
03e52840d   Kload   Init
483
484
485
486
  			$availableSpace = $availableSpace + $sizeOfDeletedVersions;
  			$versionsSize = $versionsSize - $sizeOfDeletedVersions;
  
  			// if still not enough free space we rearrange the versions from all files
6d9380f96   Cédric Dupont   Update sources OC...
487
  			if ($availableSpace <= 0) {
03e52840d   Kload   Init
488
  				$result = Storage::getAllVersions($uid);
03e52840d   Kload   Init
489
  				$allVersions = $result['all'];
6d9380f96   Cédric Dupont   Update sources OC...
490
491
492
493
494
  				foreach ($result['by_file'] as $versions) {
  					list($toDeleteNew, $size) = self::getExpireList($time, $versions);
  					$toDelete = array_merge($toDelete, $toDeleteNew);
  					$sizeOfDeletedVersions += $size;
  				}
03e52840d   Kload   Init
495
496
497
  				$availableSpace = $availableSpace + $sizeOfDeletedVersions;
  				$versionsSize = $versionsSize - $sizeOfDeletedVersions;
  			}
6d9380f96   Cédric Dupont   Update sources OC...
498
499
500
501
502
503
504
  			foreach($toDelete as $key => $path) {
  				\OC_Hook::emit('\OCP\Versions', 'preDelete', array('path' => $path));
  				self::deleteVersion($versionsFileview, $path);
  				\OC_Hook::emit('\OCP\Versions', 'delete', array('path' => $path));
  				unset($allVersions[$key]); // update array with the versions we keep
  				\OCP\Util::writeLog('files_versions', "Expire: " . $path, \OCP\Util::DEBUG);
  			}
03e52840d   Kload   Init
505
506
507
508
509
510
511
  			// Check if enough space is available after versions are rearranged.
  			// If not we delete the oldest versions until we meet the size limit for versions,
  			// but always keep the two latest versions
  			$numOfVersions = count($allVersions) -2 ;
  			$i = 0;
  			while ($availableSpace < 0 && $i < $numOfVersions) {
  				$version = current($allVersions);
6d9380f96   Cédric Dupont   Update sources OC...
512
513
  				\OC_Hook::emit('\OCP\Versions', 'preDelete', array('path' => $version['path'].'.v'.$version['version']));
  				self::deleteVersion($versionsFileview, $version['path'] . '.v' . $version['version']);
31b7f2792   Kload   Upgrade to ownclo...
514
  				\OC_Hook::emit('\OCP\Versions', 'delete', array('path' => $version['path'].'.v'.$version['version']));
6d9380f96   Cédric Dupont   Update sources OC...
515
  				\OCP\Util::writeLog('files_versions', 'running out of space! Delete oldest version: ' . $version['path'].'.v'.$version['version'] , \OCP\Util::DEBUG);
03e52840d   Kload   Init
516
517
518
519
520
521
522
523
524
525
526
527
528
  				$versionsSize -= $version['size'];
  				$availableSpace += $version['size'];
  				next($allVersions);
  				$i++;
  			}
  
  			return $versionsSize; // finally return the new size of the version history
  		}
  
  		return false;
  	}
  
  	/**
6d9380f96   Cédric Dupont   Update sources OC...
529
  	 * create recursively missing directories
03e52840d   Kload   Init
530
531
532
533
  	 * @param string $filename $path to a file
  	 * @param \OC\Files\View $view view on data/user/
  	 */
  	private static function createMissingDirectories($filename, $view) {
6d9380f96   Cédric Dupont   Update sources OC...
534
  		$dirname = \OC\Files\Filesystem::normalizePath(dirname($filename));
03e52840d   Kload   Init
535
536
537
538
539
540
541
542
543
544
545
  		$dirParts = explode('/', $dirname);
  		$dir = "/files_versions";
  		foreach ($dirParts as $part) {
  			$dir = $dir . '/' . $part;
  			if (!$view->file_exists($dir)) {
  				$view->mkdir($dir);
  			}
  		}
  	}
  
  }