aboutsummaryrefslogtreecommitdiffstats
path: root/lib/private/archive/tar.php
diff options
context:
space:
mode:
Diffstat (limited to 'lib/private/archive/tar.php')
-rw-r--r--lib/private/archive/tar.php339
1 files changed, 339 insertions, 0 deletions
diff --git a/lib/private/archive/tar.php b/lib/private/archive/tar.php
new file mode 100644
index 00000000000..a1c0535b1c3
--- /dev/null
+++ b/lib/private/archive/tar.php
@@ -0,0 +1,339 @@
+<?php
+/**
+ * Copyright (c) 2012 Robin Appelman <icewind@owncloud.com>
+ * This file is licensed under the Affero General Public License version 3 or
+ * later.
+ * See the COPYING-README file.
+ */
+
+require_once OC::$THIRDPARTYROOT . '/3rdparty/Archive/Tar.php';
+
+class OC_Archive_TAR extends OC_Archive{
+ const PLAIN=0;
+ const GZIP=1;
+ const BZIP=2;
+
+ private $fileList;
+ private $cachedHeaders;
+
+ /**
+ * @var Archive_Tar tar
+ */
+ private $tar=null;
+ private $path;
+
+ function __construct($source) {
+ $types=array(null, 'gz', 'bz');
+ $this->path=$source;
+ $this->tar=new Archive_Tar($source, $types[self::getTarType($source)]);
+ }
+
+ /**
+ * try to detect the type of tar compression
+ * @param string file
+ * @return str
+ */
+ static public function getTarType($file) {
+ if(strpos($file, '.')) {
+ $extension=substr($file, strrpos($file, '.'));
+ switch($extension) {
+ case 'gz':
+ case 'tgz':
+ return self::GZIP;
+ case 'bz':
+ case 'bz2':
+ return self::BZIP;
+ default:
+ return self::PLAIN;
+ }
+ }else{
+ return self::PLAIN;
+ }
+ }
+
+ /**
+ * add an empty folder to the archive
+ * @param string path
+ * @return bool
+ */
+ function addFolder($path) {
+ $tmpBase=OC_Helper::tmpFolder();
+ if(substr($path, -1, 1)!='/') {
+ $path.='/';
+ }
+ if($this->fileExists($path)) {
+ return false;
+ }
+ $parts=explode('/', $path);
+ $folder=$tmpBase;
+ foreach($parts as $part) {
+ $folder.='/'.$part;
+ if(!is_dir($folder)) {
+ mkdir($folder);
+ }
+ }
+ $result=$this->tar->addModify(array($tmpBase.$path), '', $tmpBase);
+ rmdir($tmpBase.$path);
+ $this->fileList=false;
+ $this->cachedHeaders=false;
+ return $result;
+ }
+ /**
+ * add a file to the archive
+ * @param string path
+ * @param string source either a local file or string data
+ * @return bool
+ */
+ function addFile($path, $source='') {
+ if($this->fileExists($path)) {
+ $this->remove($path);
+ }
+ if($source and $source[0]=='/' and file_exists($source)) {
+ $header=array();
+ $dummy='';
+ $this->tar->_openAppend();
+ $result=$this->tar->_addfile($source, $header, $dummy, $dummy, $path);
+ }else{
+ $result=$this->tar->addString($path, $source);
+ }
+ $this->fileList=false;
+ $this->cachedHeaders=false;
+ return $result;
+ }
+
+ /**
+ * rename a file or folder in the archive
+ * @param string source
+ * @param string dest
+ * @return bool
+ */
+ function rename($source, $dest) {
+ //no proper way to delete, rename entire archive, rename file and remake archive
+ $tmp=OCP\Files::tmpFolder();
+ $this->tar->extract($tmp);
+ rename($tmp.$source, $tmp.$dest);
+ $this->tar=null;
+ unlink($this->path);
+ $types=array(null, 'gz', 'bz');
+ $this->tar=new Archive_Tar($this->path, $types[self::getTarType($this->path)]);
+ $this->tar->createModify(array($tmp), '', $tmp.'/');
+ $this->fileList=false;
+ $this->cachedHeaders=false;
+ return true;
+ }
+
+ private function getHeader($file) {
+ if ( ! $this->cachedHeaders ) {
+ $this->cachedHeaders = $this->tar->listContent();
+ }
+ foreach($this->cachedHeaders as $header) {
+ if( $file == $header['filename']
+ or $file.'/' == $header['filename']
+ or '/'.$file.'/' == $header['filename']
+ or '/'.$file == $header['filename']) {
+ return $header;
+ }
+ }
+ return null;
+ }
+
+ /**
+ * get the uncompressed size of a file in the archive
+ * @param string path
+ * @return int
+ */
+ function filesize($path) {
+ $stat=$this->getHeader($path);
+ return $stat['size'];
+ }
+ /**
+ * get the last modified time of a file in the archive
+ * @param string path
+ * @return int
+ */
+ function mtime($path) {
+ $stat=$this->getHeader($path);
+ return $stat['mtime'];
+ }
+
+ /**
+ * get the files in a folder
+ * @param path
+ * @return array
+ */
+ function getFolder($path) {
+ $files=$this->getFiles();
+ $folderContent=array();
+ $pathLength=strlen($path);
+ foreach($files as $file) {
+ if($file[0]=='/') {
+ $file=substr($file, 1);
+ }
+ if(substr($file, 0, $pathLength)==$path and $file!=$path) {
+ $result=substr($file, $pathLength);
+ if($pos=strpos($result, '/')) {
+ $result=substr($result, 0, $pos+1);
+ }
+ if(array_search($result, $folderContent)===false) {
+ $folderContent[]=$result;
+ }
+ }
+ }
+ return $folderContent;
+ }
+ /**
+ * get all files in the archive
+ * @return array
+ */
+ function getFiles() {
+ if($this->fileList) {
+ return $this->fileList;
+ }
+ if ( ! $this->cachedHeaders ) {
+ $this->cachedHeaders = $this->tar->listContent();
+ }
+ $files=array();
+ foreach($this->cachedHeaders as $header) {
+ $files[]=$header['filename'];
+ }
+ $this->fileList=$files;
+ return $files;
+ }
+ /**
+ * get the content of a file
+ * @param string path
+ * @return string
+ */
+ function getFile($path) {
+ return $this->tar->extractInString($path);
+ }
+ /**
+ * extract a single file from the archive
+ * @param string path
+ * @param string dest
+ * @return bool
+ */
+ function extractFile($path, $dest) {
+ $tmp=OCP\Files::tmpFolder();
+ if(!$this->fileExists($path)) {
+ return false;
+ }
+ if($this->fileExists('/'.$path)) {
+ $success=$this->tar->extractList(array('/'.$path), $tmp);
+ }else{
+ $success=$this->tar->extractList(array($path), $tmp);
+ }
+ if($success) {
+ rename($tmp.$path, $dest);
+ }
+ OCP\Files::rmdirr($tmp);
+ return $success;
+ }
+ /**
+ * extract the archive
+ * @param string path
+ * @param string dest
+ * @return bool
+ */
+ function extract($dest) {
+ return $this->tar->extract($dest);
+ }
+ /**
+ * check if a file or folder exists in the archive
+ * @param string path
+ * @return bool
+ */
+ function fileExists($path) {
+ $files=$this->getFiles();
+ if((array_search($path, $files)!==false) or (array_search($path.'/', $files)!==false)) {
+ return true;
+ }else{
+ $folderPath=$path;
+ if(substr($folderPath, -1, 1)!='/') {
+ $folderPath.='/';
+ }
+ $pathLength=strlen($folderPath);
+ foreach($files as $file) {
+ if(strlen($file)>$pathLength and substr($file, 0, $pathLength)==$folderPath) {
+ return true;
+ }
+ }
+ }
+ if($path[0]!='/') {//not all programs agree on the use of a leading /
+ return $this->fileExists('/'.$path);
+ }else{
+ return false;
+ }
+ }
+
+ /**
+ * remove a file or folder from the archive
+ * @param string path
+ * @return bool
+ */
+ function remove($path) {
+ if(!$this->fileExists($path)) {
+ return false;
+ }
+ $this->fileList=false;
+ $this->cachedHeaders=false;
+ //no proper way to delete, extract entire archive, delete file and remake archive
+ $tmp=OCP\Files::tmpFolder();
+ $this->tar->extract($tmp);
+ OCP\Files::rmdirr($tmp.$path);
+ $this->tar=null;
+ unlink($this->path);
+ $this->reopen();
+ $this->tar->createModify(array($tmp), '', $tmp);
+ return true;
+ }
+ /**
+ * get a file handler
+ * @param string path
+ * @param string mode
+ * @return resource
+ */
+ function getStream($path, $mode) {
+ if(strrpos($path, '.')!==false) {
+ $ext=substr($path, strrpos($path, '.'));
+ }else{
+ $ext='';
+ }
+ $tmpFile=OCP\Files::tmpFile($ext);
+ if($this->fileExists($path)) {
+ $this->extractFile($path, $tmpFile);
+ }elseif($mode=='r' or $mode=='rb') {
+ return false;
+ }
+ if($mode=='r' or $mode=='rb') {
+ return fopen($tmpFile, $mode);
+ }else{
+ \OC\Files\Stream\Close::registerCallback($tmpFile, array($this, 'writeBack'));
+ self::$tempFiles[$tmpFile]=$path;
+ return fopen('close://'.$tmpFile, $mode);
+ }
+ }
+
+ private static $tempFiles=array();
+ /**
+ * write back temporary files
+ */
+ function writeBack($tmpFile) {
+ if(isset(self::$tempFiles[$tmpFile])) {
+ $this->addFile(self::$tempFiles[$tmpFile], $tmpFile);
+ unlink($tmpFile);
+ }
+ }
+
+ /**
+ * reopen the archive to ensure everything is written
+ */
+ private function reopen() {
+ if($this->tar) {
+ $this->tar->_close();
+ $this->tar=null;
+ }
+ $types=array(null, 'gz', 'bz');
+ $this->tar=new Archive_Tar($this->path, $types[self::getTarType($this->path)]);
+ }
+}