zip_stream.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt. |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Author: Piere-Alain Joye <pierre@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #ifdef HAVE_CONFIG_H
  19. # include "config.h"
  20. #endif
  21. #include "php.h"
  22. #if HAVE_ZIP
  23. #ifdef ZEND_ENGINE_2
  24. #include "php_streams.h"
  25. #include "ext/standard/file.h"
  26. #include "ext/standard/php_string.h"
  27. #include "fopen_wrappers.h"
  28. #include "php_zip.h"
  29. #include "ext/standard/url.h"
  30. struct php_zip_stream_data_t {
  31. struct zip *za;
  32. struct zip_file *zf;
  33. size_t cursor;
  34. php_stream *stream;
  35. };
  36. #define STREAM_DATA_FROM_STREAM() \
  37. struct php_zip_stream_data_t *self = (struct php_zip_stream_data_t *) stream->abstract;
  38. /* {{{ php_zip_ops_read */
  39. static size_t php_zip_ops_read(php_stream *stream, char *buf, size_t count TSRMLS_DC)
  40. {
  41. ssize_t n = 0;
  42. STREAM_DATA_FROM_STREAM();
  43. if (self->za && self->zf) {
  44. n = zip_fread(self->zf, buf, count);
  45. if (n < 0) {
  46. int ze, se;
  47. zip_file_error_get(self->zf, &ze, &se);
  48. stream->eof = 1;
  49. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Zip stream error: %s", zip_file_strerror(self->zf));
  50. return 0;
  51. }
  52. /* cast count to signed value to avoid possibly negative n
  53. * being cast to unsigned value */
  54. if (n == 0 || n < (ssize_t)count) {
  55. stream->eof = 1;
  56. } else {
  57. self->cursor += n;
  58. }
  59. }
  60. return (n < 1 ? 0 : (size_t)n);
  61. }
  62. /* }}} */
  63. /* {{{ php_zip_ops_write */
  64. static size_t php_zip_ops_write(php_stream *stream, const char *buf, size_t count TSRMLS_DC)
  65. {
  66. if (!stream) {
  67. return 0;
  68. }
  69. return count;
  70. }
  71. /* }}} */
  72. /* {{{ php_zip_ops_close */
  73. static int php_zip_ops_close(php_stream *stream, int close_handle TSRMLS_DC)
  74. {
  75. STREAM_DATA_FROM_STREAM();
  76. if (close_handle) {
  77. if (self->zf) {
  78. zip_fclose(self->zf);
  79. self->zf = NULL;
  80. }
  81. if (self->za) {
  82. zip_close(self->za);
  83. self->za = NULL;
  84. }
  85. }
  86. efree(self);
  87. stream->abstract = NULL;
  88. return EOF;
  89. }
  90. /* }}} */
  91. /* {{{ php_zip_ops_flush */
  92. static int php_zip_ops_flush(php_stream *stream TSRMLS_DC)
  93. {
  94. if (!stream) {
  95. return 0;
  96. }
  97. return 0;
  98. }
  99. /* }}} */
  100. static int php_zip_ops_stat(php_stream *stream, php_stream_statbuf *ssb TSRMLS_DC) /* {{{ */
  101. {
  102. struct zip_stat sb;
  103. const char *path = stream->orig_path;
  104. size_t path_len = strlen(stream->orig_path);
  105. char *file_basename;
  106. size_t file_basename_len;
  107. char file_dirname[MAXPATHLEN];
  108. struct zip *za;
  109. char *fragment;
  110. size_t fragment_len;
  111. int err;
  112. fragment = strchr(path, '#');
  113. if (!fragment) {
  114. return -1;
  115. }
  116. if (strncasecmp("zip://", path, 6) == 0) {
  117. path += 6;
  118. }
  119. fragment_len = strlen(fragment);
  120. if (fragment_len < 1) {
  121. return -1;
  122. }
  123. path_len = strlen(path);
  124. if (path_len >= MAXPATHLEN) {
  125. return -1;
  126. }
  127. memcpy(file_dirname, path, path_len - fragment_len);
  128. file_dirname[path_len - fragment_len] = '\0';
  129. php_basename((char *)path, path_len - fragment_len, NULL, 0, &file_basename, &file_basename_len TSRMLS_CC);
  130. fragment++;
  131. if (ZIP_OPENBASEDIR_CHECKPATH(file_dirname)) {
  132. efree(file_basename);
  133. return -1;
  134. }
  135. za = zip_open(file_dirname, ZIP_CREATE, &err);
  136. if (za) {
  137. memset(ssb, 0, sizeof(php_stream_statbuf));
  138. if (zip_stat(za, fragment, ZIP_FL_NOCASE, &sb) != 0) {
  139. efree(file_basename);
  140. return -1;
  141. }
  142. zip_close(za);
  143. if (path[path_len-1] != '/') {
  144. ssb->sb.st_size = sb.size;
  145. ssb->sb.st_mode |= S_IFREG; /* regular file */
  146. } else {
  147. ssb->sb.st_size = 0;
  148. ssb->sb.st_mode |= S_IFDIR; /* regular directory */
  149. }
  150. ssb->sb.st_mtime = sb.mtime;
  151. ssb->sb.st_atime = sb.mtime;
  152. ssb->sb.st_ctime = sb.mtime;
  153. ssb->sb.st_nlink = 1;
  154. ssb->sb.st_rdev = -1;
  155. #ifndef PHP_WIN32
  156. ssb->sb.st_blksize = -1;
  157. ssb->sb.st_blocks = -1;
  158. #endif
  159. ssb->sb.st_ino = -1;
  160. }
  161. efree(file_basename);
  162. return 0;
  163. }
  164. /* }}} */
  165. php_stream_ops php_stream_zipio_ops = {
  166. php_zip_ops_write, php_zip_ops_read,
  167. php_zip_ops_close, php_zip_ops_flush,
  168. "zip",
  169. NULL, /* seek */
  170. NULL, /* cast */
  171. php_zip_ops_stat, /* stat */
  172. NULL /* set_option */
  173. };
  174. /* {{{ php_stream_zip_open */
  175. php_stream *php_stream_zip_open(const char *filename, const char *path, const char *mode STREAMS_DC TSRMLS_DC)
  176. {
  177. struct zip_file *zf = NULL;
  178. int err = 0;
  179. php_stream *stream = NULL;
  180. struct php_zip_stream_data_t *self;
  181. struct zip *stream_za;
  182. if (strncmp(mode,"r", strlen("r")) != 0) {
  183. return NULL;
  184. }
  185. if (filename) {
  186. if (ZIP_OPENBASEDIR_CHECKPATH(filename)) {
  187. return NULL;
  188. }
  189. /* duplicate to make the stream za independent (esp. for MSHUTDOWN) */
  190. stream_za = zip_open(filename, ZIP_CREATE, &err);
  191. if (!stream_za) {
  192. return NULL;
  193. }
  194. zf = zip_fopen(stream_za, path, 0);
  195. if (zf) {
  196. self = emalloc(sizeof(*self));
  197. self->za = stream_za;
  198. self->zf = zf;
  199. self->stream = NULL;
  200. self->cursor = 0;
  201. stream = php_stream_alloc(&php_stream_zipio_ops, self, NULL, mode);
  202. stream->orig_path = estrdup(path);
  203. } else {
  204. zip_close(stream_za);
  205. }
  206. }
  207. if (!stream) {
  208. return NULL;
  209. } else {
  210. return stream;
  211. }
  212. }
  213. /* }}} */
  214. /* {{{ php_stream_zip_opener */
  215. php_stream *php_stream_zip_opener(php_stream_wrapper *wrapper,
  216. const char *path,
  217. const char *mode,
  218. int options,
  219. char **opened_path,
  220. php_stream_context *context STREAMS_DC TSRMLS_DC)
  221. {
  222. size_t path_len;
  223. char *file_basename;
  224. size_t file_basename_len;
  225. char file_dirname[MAXPATHLEN];
  226. struct zip *za;
  227. struct zip_file *zf = NULL;
  228. char *fragment;
  229. size_t fragment_len;
  230. int err;
  231. php_stream *stream = NULL;
  232. struct php_zip_stream_data_t *self;
  233. fragment = strchr(path, '#');
  234. if (!fragment) {
  235. return NULL;
  236. }
  237. if (strncasecmp("zip://", path, 6) == 0) {
  238. path += 6;
  239. }
  240. fragment_len = strlen(fragment);
  241. if (fragment_len < 1) {
  242. return NULL;
  243. }
  244. path_len = strlen(path);
  245. if (path_len >= MAXPATHLEN || mode[0] != 'r') {
  246. return NULL;
  247. }
  248. memcpy(file_dirname, path, path_len - fragment_len);
  249. file_dirname[path_len - fragment_len] = '\0';
  250. php_basename(path, path_len - fragment_len, NULL, 0, &file_basename, &file_basename_len TSRMLS_CC);
  251. fragment++;
  252. if (ZIP_OPENBASEDIR_CHECKPATH(file_dirname)) {
  253. efree(file_basename);
  254. return NULL;
  255. }
  256. za = zip_open(file_dirname, ZIP_CREATE, &err);
  257. if (za) {
  258. zf = zip_fopen(za, fragment, 0);
  259. if (zf) {
  260. self = emalloc(sizeof(*self));
  261. self->za = za;
  262. self->zf = zf;
  263. self->stream = NULL;
  264. self->cursor = 0;
  265. stream = php_stream_alloc(&php_stream_zipio_ops, self, NULL, mode);
  266. if (opened_path) {
  267. *opened_path = estrdup(path);
  268. }
  269. } else {
  270. zip_close(za);
  271. }
  272. }
  273. efree(file_basename);
  274. if (!stream) {
  275. return NULL;
  276. } else {
  277. return stream;
  278. }
  279. }
  280. /* }}} */
  281. static php_stream_wrapper_ops zip_stream_wops = {
  282. php_stream_zip_opener,
  283. NULL, /* close */
  284. NULL, /* fstat */
  285. NULL, /* stat */
  286. NULL, /* opendir */
  287. "zip wrapper",
  288. NULL, /* unlink */
  289. NULL, /* rename */
  290. NULL, /* mkdir */
  291. NULL /* rmdir */
  292. };
  293. php_stream_wrapper php_stream_zip_wrapper = {
  294. &zip_stream_wops,
  295. NULL,
  296. 0 /* is_url */
  297. };
  298. #endif /* ZEND_ENGINE_2 */
  299. #endif /* HAVE_ZIP */