Skip to content

Commit 859532b

Browse files
committed
changes
1 parent 347350f commit 859532b

12 files changed

+734
-429
lines changed

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -34,3 +34,5 @@ nosetests.xml
3434
.mr.developer.cfg
3535
.project
3636
.pydevproject
37+
38+
*.zip

.travis.yml

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
language: python
2+
python:
3+
- "2.6"
4+
- "2.7"
5+
- "3.2"
6+
- "3.3"
7+
- "pypy"
8+
install:
9+
- "pip install ."
10+
script: nosetests

README.markdown

+64-9
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
11

2-
# ZipStream
2+
# python-zipstream
33

4-
zipstream.py is a zip archive generator based on zipfile.py. It was created to
5-
generate a zip file on-the-fly for download in a web.py (http://webpy.org/)
6-
application. This is beneficial for when you want to provide a downloadable
7-
archive of a large collection of regular files, which would be infeasible to
4+
[![Build Status](https://travis-ci.org/allanlei/python-zipstream.png?branch=master)](https://travis-ci.org/allanlei/python-zipstream)
5+
[![Coverage Status](https://coveralls.io/repos/allanlei/python-zipstream/badge.png)](https://coveralls.io/r/allanlei/python-zipstream)
6+
7+
zipstream.py is a zip archive generator based on python 3.3's zipfile.py. It was created to
8+
generate a zip file generator for streaming (ie web apps). This is beneficial for when you
9+
want to provide a downloadable archive of a large collection of regular files, which would be infeasible to
810
generate the archive prior to downloading.
911

1012
The archive is generated as an iterator of strings, which, when joined, form
@@ -37,10 +39,63 @@ archives.
3739

3840
## Requirements
3941

40-
* Python >=2.6
42+
* Python 2.6, 2.7, 3.2, 3.3, pypy
43+
44+
## Examples
45+
46+
### flask
47+
48+
```python
49+
from flask import Response
50+
51+
@app.route('/package.zip', methods=['GET'], endpoint='zipball')
52+
def zipball():
53+
def generator():
54+
z = ZipStream(mode='w', compression=ZIP_DEFLATED)
55+
56+
z.write('/path/to/file')
4157

42-
## License
58+
for chunk in z:
59+
yield chunk
4360

44-
This library was created by SpiderOak, Inc. and is released under the GPLv3.
45-
Copyright 2008-2013 SpiderOak Inc.
61+
response = Response(generator(), mimetype='application/zip')
62+
response.headers['Content-Disposition'] = 'attachment; filename={}'.format('files.zip')
63+
return response
64+
65+
# or
66+
67+
@app.route('/package.zip', methods=['GET'], endpoint='zipball')
68+
def zipball():
69+
z = ZipStream(mode='w', compression=ZIP_DEFLATED)
70+
z.write('/path/to/file')
71+
72+
response = Response(z, mimetype='application/zip')
73+
response.headers['Content-Disposition'] = 'attachment; filename={}'.format('files.zip')
74+
return response
75+
```
76+
77+
### django 1.5+
78+
79+
```python
80+
from django.http import StreamingHttpResponse
4681

82+
def zipball(request):
83+
z = ZipStream(mode='w', compression=ZIP_DEFLATED)
84+
z.write('/path/to/file')
85+
86+
response = StreamingHttpResponse(z, mimetype='application/zip')
87+
response['Content-Disposition'] = 'attachment; filename={}'.format('files.zip')
88+
return response
89+
```
90+
91+
### webpy
92+
93+
```python
94+
def GET(self):
95+
path = '/path/to/dir/of/files'
96+
zip_filename = 'files.zip'
97+
web.header('Content-type' , 'application/zip')
98+
web.header('Content-Disposition', 'attachment; filename="%s"' % (
99+
zip_filename,))
100+
return ZipStream(path)
101+
```

example.py

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
import os
2+
import zipstream
3+
import zipfile
4+
5+
6+
f = open('test.zip', 'wb')
7+
8+
with zipstream.ZipFile(mode='w', compression=zipstream.ZIP_DEFLATED) as z:
9+
z.write('LICENSE')
10+
z.write('LICENSE', arcname='stuff/LICENSE')
11+
12+
for root, directories, files in os.walk('zipstream'):
13+
for filename in files:
14+
path = os.path.join(root, filename)
15+
z.write(path, path)
16+
17+
with open('test.zip', 'wb') as f:
18+
for chunk in z:
19+
f.write(chunk)
20+
21+
f.close()
22+
23+
24+
with zipfile.ZipFile('test.zip') as z:
25+
z.testzip()

setup.py

+13-8
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,18 @@
1-
#!/usr/bin/env python
1+
# -*- coding: utf-8 -*-
2+
from setuptools import setup, find_packages
3+
import zipstream
24

3-
from distutils.core import setup
45

56
setup(
67
name='zipstream',
7-
version='1.0.1',
8-
description='SpiderOak ZipStream Module',
9-
author='SpiderOak Team',
10-
author_email='[email protected]',
11-
url='http://www.spideroak.com',
12-
py_modules=['zipstream'],
8+
version=zipstream.__version__,
9+
description='Zipfile generator',
10+
author='Allan Lei',
11+
author_email='[email protected]',
12+
url='https://github.com/allanlei/python-zipstream',
13+
packages=find_packages(),
14+
keywords='zip streaming',
15+
16+
test_suite='nose.collector',
17+
tests_require = ['nose'],
1318
)

tests/__init__.py

Whitespace-only changes.

tests/test_pointerio.py

+80
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
# -*- coding: utf-8 -*-
2+
from __future__ import unicode_literals
3+
4+
import unittest
5+
import zipstream
6+
7+
8+
class PointerIOTestCase(unittest.TestCase):
9+
def test_init_no_args(self):
10+
zipstream.PointerIO()
11+
12+
def test_init_mode(self):
13+
try:
14+
zipstream.PointerIO('wb')
15+
except Exception as err:
16+
self.fail(err)
17+
18+
for mode in ['w', 'r', 'rb', 'a', 'ab']:
19+
self.assertRaises(Exception, zipstream.PointerIO, mode=mode)
20+
21+
for mode in ['w', 'wb''r', 'rb', 'a', 'ab']:
22+
self.assertRaises(Exception, zipstream.PointerIO, mode=mode + '+')
23+
24+
def test_has_fileobj_attrs(self):
25+
fileobj = zipstream.PointerIO()
26+
27+
self.assertTrue(hasattr(fileobj, 'write'))
28+
self.assertTrue(hasattr(fileobj, 'close'))
29+
self.assertTrue(hasattr(fileobj, 'tell'))
30+
31+
def test_write_bytes(self):
32+
fileobj = zipstream.PointerIO()
33+
data = b'Im a little tea pot'
34+
try:
35+
fileobj.write(data)
36+
except Exception as err:
37+
self.fail(err)
38+
self.assertEqual(fileobj.tell(), 19)
39+
40+
def test_write_unicode(self):
41+
fileobj = zipstream.PointerIO()
42+
data = 'Im a little tea pot'
43+
try:
44+
fileobj.write(data)
45+
except Exception as err:
46+
self.fail(err)
47+
self.assertEqual(fileobj.tell(), 19)
48+
49+
50+
fileobj = zipstream.PointerIO()
51+
data = '幋 儳鑤 寱懤擨 拻敁柧'
52+
try:
53+
fileobj.write(data)
54+
except Exception as err:
55+
self.fail(err)
56+
self.assertEqual(fileobj.tell(), 30)
57+
58+
def test_write_non_string_type(self):
59+
fileobj = zipstream.PointerIO()
60+
data = None
61+
self.assertRaises(TypeError, fileobj.write, data)
62+
63+
fileobj = zipstream.PointerIO()
64+
data = []
65+
self.assertRaises(TypeError, fileobj.write, data)
66+
67+
fileobj = zipstream.PointerIO()
68+
data = tuple()
69+
self.assertRaises(TypeError, fileobj.write, data)
70+
71+
fileobj = zipstream.PointerIO()
72+
data = 1
73+
self.assertRaises(TypeError, fileobj.write, data)
74+
75+
fileobj = zipstream.PointerIO()
76+
data = 1.00
77+
self.assertRaises(TypeError, fileobj.write, data)
78+
79+
if __name__ == '__main__':
80+
unittest.main()

tests/test_zipstream.py

+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
# -*- coding: utf-8 -*-
2+
from __future__ import unicode_literals, print_function
3+
4+
import os
5+
import tempfile
6+
import unittest
7+
import zipstream
8+
import zipfile
9+
10+
11+
class ZipInfoTestCase(unittest.TestCase):
12+
pass
13+
14+
15+
class ZipStreamTestCase(unittest.TestCase):
16+
def setUp(self):
17+
self.fileobjs = [
18+
tempfile.NamedTemporaryFile(delete=False, suffix='.txt'),
19+
tempfile.NamedTemporaryFile(delete=False, suffix='.py'),
20+
]
21+
22+
def tearDown(self):
23+
for fileobj in self.fileobjs:
24+
fileobj.close()
25+
os.remove(fileobj.name)
26+
27+
def test_init_no_args(self):
28+
zipstream.ZipFile()
29+
30+
def test_init_mode(self):
31+
try:
32+
zipstream.ZipFile(mode='w')
33+
except Exception as err:
34+
self.fail(err)
35+
36+
for mode in ['wb', 'r', 'rb', 'a', 'ab']:
37+
self.assertRaises(Exception, zipstream.ZipFile, mode=mode)
38+
39+
for mode in ['wb', 'r', 'rb', 'a', 'ab']:
40+
self.assertRaises(Exception, zipstream.ZipFile, mode=mode + '+')
41+
42+
def test_write_file(self):
43+
z = zipstream.ZipFile(mode='w')
44+
for fileobj in self.fileobjs:
45+
z.write(fileobj.name)
46+
47+
f = tempfile.NamedTemporaryFile(suffix='zip', delete=False)
48+
for chunk in z:
49+
f.write(chunk)
50+
f.close()
51+
52+
z2 = zipfile.ZipFile(f.name, 'r')
53+
z2.testzip()
54+
55+
os.remove(f.name)
56+
57+
58+
if __name__ == '__main__':
59+
unittest.main()

tox.ini

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
[tox]
2+
envlist = py26, py27, py32, py33, pypy
3+
4+
[testenv]
5+
deps=nose
6+
commands = nosetests {posargs}

0 commit comments

Comments
 (0)