Merge remote-tracking branch 'use_pip/use-pip'
# Conflicts: # cps.py # readme.md # vendor/wand/api.py # vendor/wand/drawing.py # vendor/wand/exceptions.py # vendor/wand/image.py # vendor/wand/version.py
This commit is contained in:
		
						commit
						acac0570e5
					
				|  | @ -18,9 +18,8 @@ from sqlalchemy.exc import IntegrityError | |||
| from sqlalchemy import __version__ as sqlalchemyVersion | ||||
| from math import ceil | ||||
| from flask_login import LoginManager, login_user, logout_user, login_required, current_user | ||||
| from flask_login import __version__ as flask_loginVersion | ||||
| from flask_principal import Principal, Identity, AnonymousIdentity, identity_changed | ||||
| from flask_login import __version__ as flask_principalVersion | ||||
| from flask_principal import __version__ as flask_principalVersion | ||||
| from flask_babel import Babel | ||||
| from flask_babel import gettext as _ | ||||
| import requests | ||||
|  | @ -48,6 +47,11 @@ from shutil import move, copyfile | |||
| from tornado.ioloop import IOLoop | ||||
| from tornado import version as tornadoVersion | ||||
| 
 | ||||
| try: | ||||
|     from flask_login import __version__ as flask_loginVersion | ||||
| except ImportError, e: | ||||
|     from flask_login.__about__ import __version__ as flask_loginVersion | ||||
| 
 | ||||
| try: | ||||
|     from wand.image import Image | ||||
| 
 | ||||
|  |  | |||
							
								
								
									
										26
									
								
								readme.md
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								readme.md
									
									
									
									
									
								
							|  | @ -1,4 +1,4 @@ | |||
| ##About | ||||
| ## About | ||||
| 
 | ||||
| Calibre Web is a web app providing a clean interface for browsing, reading and downloading eBooks using an existing [Calibre](https://calibre-ebook.com) database. | ||||
| 
 | ||||
|  | @ -6,7 +6,8 @@ Calibre Web is a web app providing a clean interface for browsing, reading and d | |||
| 
 | ||||
|  | ||||
| 
 | ||||
| ##Features | ||||
| ## Features | ||||
| 
 | ||||
| - Bootstrap 3 HTML5 interface | ||||
| - full graphical setup | ||||
| - User management | ||||
|  | @ -28,13 +29,14 @@ Calibre Web is a web app providing a clean interface for browsing, reading and d | |||
| 
 | ||||
| ## Quick start | ||||
| 
 | ||||
| 1. Execute the command: `python cps.py` (or `nohup python cps.py` - recommended if you want to exit the terminal window) | ||||
| 2. Point your browser to `http://localhost:8083` or `http://localhost:8083/opds` for the OPDS catalog  | ||||
| 3. Set `Location of Calibre database` to the path of the folder where your Calibre library (metadata.db) lives, push "submit" button | ||||
| 4. Go to Login page | ||||
| 1. Install required dependencies by executing `pip install -r requirements.txt` | ||||
| 2. Execute the command: `python cps.py` (or `nohup python cps.py` - recommended if you want to exit the terminal window) | ||||
| 3. Point your browser to `http://localhost:8083` or `http://localhost:8083/opds` for the OPDS catalog  | ||||
| 4. Set `Location of Calibre database` to the path of the folder where your Calibre library (metadata.db) lives, push "submit" button | ||||
| 5. Go to Login page | ||||
| 
 | ||||
| **Default admin login:**     | ||||
| *Username:* admin    | ||||
| **Default admin login:** | ||||
| *Username:* admin | ||||
| *Password:* admin123 | ||||
| 
 | ||||
| ## Runtime Configuration Options | ||||
|  | @ -56,10 +58,10 @@ Tick to enable uploading of PDF, epub, FB2. This requires the imagemagick librar | |||
| ## Requirements | ||||
| 
 | ||||
| Python 2.7+ | ||||
|       | ||||
| Optionally, to enable on-the-fly conversion from EPUB to MOBI when using the send-to-kindle feature:      | ||||
| 
 | ||||
| [Download](http://www.amazon.com/gp/feature.html?docId=1000765211) Amazon's KindleGen tool for your platform and place the binary named as `kindlegen` in the `vendor` folder.  | ||||
| Optionally, to enable on-the-fly conversion from EPUB to MOBI when using the send-to-kindle feature: | ||||
| 
 | ||||
| [Download](http://www.amazon.com/gp/feature.html?docId=1000765211) Amazon's KindleGen tool for your platform and place the binary named as `kindlegen` in the `vendor` folder. | ||||
| 
 | ||||
| ## Docker image | ||||
| 
 | ||||
|  | @ -131,4 +133,4 @@ Replace the user and ExecStart with your user and foldernames. | |||
| 
 | ||||
| `sudo systemctl enable cps.service` | ||||
| 
 | ||||
| enables the service.  | ||||
| enables the service. | ||||
|  |  | |||
							
								
								
									
										12
									
								
								requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								requirements.txt
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,12 @@ | |||
| Babel>=1.3 | ||||
| Flask>=0.11 | ||||
| Flask-Babel==0.11.1 | ||||
| Flask-Login>=0.3.2 | ||||
| Flask-Principal>=0.3.2 | ||||
| iso-639>=0.4.5 | ||||
| PyPDF2==1.26.0 | ||||
| pytz>=2016.10 | ||||
| requests>=2.11.1 | ||||
| SQLAlchemy>=0.8.4 | ||||
| tornado>=4.4.2 | ||||
| Wand>=0.4.4 | ||||
							
								
								
									
										22
									
								
								vendor/LICENSE_flask_login
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								vendor/LICENSE_flask_login
									
									
									
									
										vendored
									
									
								
							|  | @ -1,22 +0,0 @@ | |||
| Copyright (c) 2011 Matthew Frazier | ||||
| 
 | ||||
| Permission is hereby granted, free of charge, to any person | ||||
| obtaining a copy of this software and associated documentation | ||||
| files (the "Software"), to deal in the Software without | ||||
| restriction, including without limitation the rights to use, | ||||
| copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the | ||||
| Software is furnished to do so, subject to the following | ||||
| conditions: | ||||
| 
 | ||||
| The above copyright notice and this permission notice shall be | ||||
| included in all copies or substantial portions of the Software. | ||||
| 
 | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||||
| EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES | ||||
| OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||||
| NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT | ||||
| HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | ||||
| WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | ||||
| FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR | ||||
| OTHER DEALINGS IN THE SOFTWARE. | ||||
							
								
								
									
										22
									
								
								vendor/LICENSE_flask_principal
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								vendor/LICENSE_flask_principal
									
									
									
									
										vendored
									
									
								
							|  | @ -1,22 +0,0 @@ | |||
| Copyright (c) 2012 Ali Afshar | ||||
| 
 | ||||
| Permission is hereby granted, free of charge, to any person | ||||
| obtaining a copy of this software and associated documentation | ||||
| files (the "Software"), to deal in the Software without | ||||
| restriction, including without limitation the rights to use, | ||||
| copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the | ||||
| Software is furnished to do so, subject to the following | ||||
| conditions: | ||||
| 
 | ||||
| The above copyright notice and this permission notice shall be | ||||
| included in all copies or substantial portions of the Software. | ||||
| 
 | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | ||||
| EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES | ||||
| OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | ||||
| NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT | ||||
| HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, | ||||
| WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | ||||
| FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR | ||||
| OTHER DEALINGS IN THE SOFTWARE. | ||||
							
								
								
									
										31
									
								
								vendor/LICENSE_itsdangerous
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										31
									
								
								vendor/LICENSE_itsdangerous
									
									
									
									
										vendored
									
									
								
							|  | @ -1,31 +0,0 @@ | |||
| Copyright (c) 2011 by Armin Ronacher and the Django Software Foundation. | ||||
| 
 | ||||
| Some rights reserved. | ||||
| 
 | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are | ||||
| met: | ||||
| 
 | ||||
|     * Redistributions of source code must retain the above copyright | ||||
|       notice, this list of conditions and the following disclaimer. | ||||
| 
 | ||||
|     * Redistributions in binary form must reproduce the above | ||||
|       copyright notice, this list of conditions and the following | ||||
|       disclaimer in the documentation and/or other materials provided | ||||
|       with the distribution. | ||||
| 
 | ||||
|     * The names of the contributors may not be used to endorse or | ||||
|       promote products derived from this software without specific | ||||
|       prior written permission. | ||||
| 
 | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | ||||
| "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | ||||
| LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | ||||
| A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | ||||
| OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | ||||
| SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | ||||
| LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | ||||
| DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | ||||
| THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||||
| (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
							
								
								
									
										5
									
								
								vendor/PyPDF2/__init__.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								vendor/PyPDF2/__init__.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,5 +0,0 @@ | |||
| from .pdf import PdfFileReader, PdfFileWriter | ||||
| from .merger import PdfFileMerger | ||||
| from .pagerange import PageRange, parse_filename_page_ranges | ||||
| from ._version import __version__ | ||||
| __all__ = ["pdf", "PdfFileMerger"] | ||||
							
								
								
									
										1
									
								
								vendor/PyPDF2/_version.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								vendor/PyPDF2/_version.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1 +0,0 @@ | |||
| __version__ = '1.26.0' | ||||
							
								
								
									
										362
									
								
								vendor/PyPDF2/filters.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										362
									
								
								vendor/PyPDF2/filters.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,362 +0,0 @@ | |||
| # vim: sw=4:expandtab:foldmethod=marker | ||||
| # | ||||
| # Copyright (c) 2006, Mathieu Fenniak | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without | ||||
| # modification, are permitted provided that the following conditions are | ||||
| # met: | ||||
| # | ||||
| # * Redistributions of source code must retain the above copyright notice, | ||||
| # this list of conditions and the following disclaimer. | ||||
| # * Redistributions in binary form must reproduce the above copyright notice, | ||||
| # this list of conditions and the following disclaimer in the documentation | ||||
| # and/or other materials provided with the distribution. | ||||
| # * The name of the author may not be used to endorse or promote products | ||||
| # derived from this software without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||
| # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE | ||||
| # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | ||||
| # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | ||||
| # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||
| # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | ||||
| # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | ||||
| # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | ||||
| # POSSIBILITY OF SUCH DAMAGE. | ||||
| 
 | ||||
| 
 | ||||
| """ | ||||
| Implementation of stream filters for PDF. | ||||
| """ | ||||
| __author__ = "Mathieu Fenniak" | ||||
| __author_email__ = "biziqe@mathieu.fenniak.net" | ||||
| 
 | ||||
| from .utils import PdfReadError, ord_, chr_ | ||||
| from sys import version_info | ||||
| if version_info < ( 3, 0 ): | ||||
|     from cStringIO import StringIO | ||||
| else: | ||||
|     from io import StringIO | ||||
|     import struct | ||||
| 
 | ||||
| try: | ||||
|     import zlib | ||||
| 
 | ||||
|     def decompress(data): | ||||
|         return zlib.decompress(data) | ||||
| 
 | ||||
|     def compress(data): | ||||
|         return zlib.compress(data) | ||||
| 
 | ||||
| except ImportError: | ||||
|     # Unable to import zlib.  Attempt to use the System.IO.Compression | ||||
|     # library from the .NET framework. (IronPython only) | ||||
|     import System | ||||
|     from System import IO, Collections, Array | ||||
| 
 | ||||
|     def _string_to_bytearr(buf): | ||||
|         retval = Array.CreateInstance(System.Byte, len(buf)) | ||||
|         for i in range(len(buf)): | ||||
|             retval[i] = ord(buf[i]) | ||||
|         return retval | ||||
| 
 | ||||
|     def _bytearr_to_string(bytes): | ||||
|         retval = "" | ||||
|         for i in range(bytes.Length): | ||||
|             retval += chr(bytes[i]) | ||||
|         return retval | ||||
| 
 | ||||
|     def _read_bytes(stream): | ||||
|         ms = IO.MemoryStream() | ||||
|         buf = Array.CreateInstance(System.Byte, 2048) | ||||
|         while True: | ||||
|             bytes = stream.Read(buf, 0, buf.Length) | ||||
|             if bytes == 0: | ||||
|                 break | ||||
|             else: | ||||
|                 ms.Write(buf, 0, bytes) | ||||
|         retval = ms.ToArray() | ||||
|         ms.Close() | ||||
|         return retval | ||||
| 
 | ||||
|     def decompress(data): | ||||
|         bytes = _string_to_bytearr(data) | ||||
|         ms = IO.MemoryStream() | ||||
|         ms.Write(bytes, 0, bytes.Length) | ||||
|         ms.Position = 0  # fseek 0 | ||||
|         gz = IO.Compression.DeflateStream(ms, IO.Compression.CompressionMode.Decompress) | ||||
|         bytes = _read_bytes(gz) | ||||
|         retval = _bytearr_to_string(bytes) | ||||
|         gz.Close() | ||||
|         return retval | ||||
| 
 | ||||
|     def compress(data): | ||||
|         bytes = _string_to_bytearr(data) | ||||
|         ms = IO.MemoryStream() | ||||
|         gz = IO.Compression.DeflateStream(ms, IO.Compression.CompressionMode.Compress, True) | ||||
|         gz.Write(bytes, 0, bytes.Length) | ||||
|         gz.Close() | ||||
|         ms.Position = 0 # fseek 0 | ||||
|         bytes = ms.ToArray() | ||||
|         retval = _bytearr_to_string(bytes) | ||||
|         ms.Close() | ||||
|         return retval | ||||
| 
 | ||||
| 
 | ||||
| class FlateDecode(object): | ||||
|     def decode(data, decodeParms): | ||||
|         data = decompress(data) | ||||
|         predictor = 1 | ||||
|         if decodeParms: | ||||
|             try: | ||||
|                 predictor = decodeParms.get("/Predictor", 1) | ||||
|             except AttributeError: | ||||
|                 pass    # usually an array with a null object was read | ||||
| 
 | ||||
|         # predictor 1 == no predictor | ||||
|         if predictor != 1: | ||||
|             columns = decodeParms["/Columns"] | ||||
|             # PNG prediction: | ||||
|             if predictor >= 10 and predictor <= 15: | ||||
|                 output = StringIO() | ||||
|                 # PNG prediction can vary from row to row | ||||
|                 rowlength = columns + 1 | ||||
|                 assert len(data) % rowlength == 0 | ||||
|                 prev_rowdata = (0,) * rowlength | ||||
|                 for row in range(len(data) // rowlength): | ||||
|                     rowdata = [ord_(x) for x in data[(row*rowlength):((row+1)*rowlength)]] | ||||
|                     filterByte = rowdata[0] | ||||
|                     if filterByte == 0: | ||||
|                         pass | ||||
|                     elif filterByte == 1: | ||||
|                         for i in range(2, rowlength): | ||||
|                             rowdata[i] = (rowdata[i] + rowdata[i-1]) % 256 | ||||
|                     elif filterByte == 2: | ||||
|                         for i in range(1, rowlength): | ||||
|                             rowdata[i] = (rowdata[i] + prev_rowdata[i]) % 256 | ||||
|                     else: | ||||
|                         # unsupported PNG filter | ||||
|                         raise PdfReadError("Unsupported PNG filter %r" % filterByte) | ||||
|                     prev_rowdata = rowdata | ||||
|                     output.write(''.join([chr(x) for x in rowdata[1:]])) | ||||
|                 data = output.getvalue() | ||||
|             else: | ||||
|                 # unsupported predictor | ||||
|                 raise PdfReadError("Unsupported flatedecode predictor %r" % predictor) | ||||
|         return data | ||||
|     decode = staticmethod(decode) | ||||
| 
 | ||||
|     def encode(data): | ||||
|         return compress(data) | ||||
|     encode = staticmethod(encode) | ||||
| 
 | ||||
| 
 | ||||
| class ASCIIHexDecode(object): | ||||
|     def decode(data, decodeParms=None): | ||||
|         retval = "" | ||||
|         char = "" | ||||
|         x = 0 | ||||
|         while True: | ||||
|             c = data[x] | ||||
|             if c == ">": | ||||
|                 break | ||||
|             elif c.isspace(): | ||||
|                 x += 1 | ||||
|                 continue | ||||
|             char += c | ||||
|             if len(char) == 2: | ||||
|                 retval += chr(int(char, base=16)) | ||||
|                 char = "" | ||||
|             x += 1 | ||||
|         assert char == "" | ||||
|         return retval | ||||
|     decode = staticmethod(decode) | ||||
| 
 | ||||
| 
 | ||||
| class LZWDecode(object): | ||||
|     """Taken from: | ||||
|     http://www.java2s.com/Open-Source/Java-Document/PDF/PDF-Renderer/com/sun/pdfview/decode/LZWDecode.java.htm | ||||
|     """ | ||||
|     class decoder(object): | ||||
|         def __init__(self, data): | ||||
|             self.STOP=257 | ||||
|             self.CLEARDICT=256 | ||||
|             self.data=data | ||||
|             self.bytepos=0 | ||||
|             self.bitpos=0 | ||||
|             self.dict=[""]*4096 | ||||
|             for i in range(256): | ||||
|                 self.dict[i]=chr(i) | ||||
|             self.resetDict() | ||||
| 
 | ||||
|         def resetDict(self): | ||||
|             self.dictlen=258 | ||||
|             self.bitspercode=9 | ||||
| 
 | ||||
|         def nextCode(self): | ||||
|             fillbits=self.bitspercode | ||||
|             value=0 | ||||
|             while fillbits>0 : | ||||
|                 if self.bytepos >= len(self.data): | ||||
|                     return -1 | ||||
|                 nextbits=ord(self.data[self.bytepos]) | ||||
|                 bitsfromhere=8-self.bitpos | ||||
|                 if bitsfromhere>fillbits: | ||||
|                     bitsfromhere=fillbits | ||||
|                 value |= (((nextbits >> (8-self.bitpos-bitsfromhere)) & | ||||
|                            (0xff >> (8-bitsfromhere))) << | ||||
|                           (fillbits-bitsfromhere)) | ||||
|                 fillbits -= bitsfromhere | ||||
|                 self.bitpos += bitsfromhere | ||||
|                 if self.bitpos >=8: | ||||
|                     self.bitpos=0 | ||||
|                     self.bytepos = self.bytepos+1 | ||||
|             return value | ||||
| 
 | ||||
|         def decode(self): | ||||
|             """ algorithm derived from: | ||||
|             http://www.rasip.fer.hr/research/compress/algorithms/fund/lz/lzw.html | ||||
|             and the PDFReference | ||||
|             """ | ||||
|             cW = self.CLEARDICT; | ||||
|             baos="" | ||||
|             while True: | ||||
|                 pW = cW; | ||||
|                 cW = self.nextCode(); | ||||
|                 if cW == -1: | ||||
|                     raise PdfReadError("Missed the stop code in LZWDecode!") | ||||
|                 if cW == self.STOP: | ||||
|                     break; | ||||
|                 elif cW == self.CLEARDICT: | ||||
|                     self.resetDict(); | ||||
|                 elif pW == self.CLEARDICT: | ||||
|                     baos+=self.dict[cW] | ||||
|                 else: | ||||
|                     if cW < self.dictlen: | ||||
|                         baos += self.dict[cW] | ||||
|                         p=self.dict[pW]+self.dict[cW][0] | ||||
|                         self.dict[self.dictlen]=p | ||||
|                         self.dictlen+=1 | ||||
|                     else: | ||||
|                         p=self.dict[pW]+self.dict[pW][0] | ||||
|                         baos+=p | ||||
|                         self.dict[self.dictlen] = p; | ||||
|                         self.dictlen+=1 | ||||
|                     if (self.dictlen >= (1 << self.bitspercode) - 1 and | ||||
|                         self.bitspercode < 12): | ||||
|                         self.bitspercode+=1 | ||||
|             return baos | ||||
| 
 | ||||
|     @staticmethod | ||||
|     def decode(data,decodeParams=None): | ||||
|         return LZWDecode.decoder(data).decode() | ||||
| 
 | ||||
| 
 | ||||
| class ASCII85Decode(object): | ||||
|     def decode(data, decodeParms=None): | ||||
|         if version_info < ( 3, 0 ): | ||||
|             retval = "" | ||||
|             group = [] | ||||
|             x = 0 | ||||
|             hitEod = False | ||||
|             # remove all whitespace from data | ||||
|             data = [y for y in data if not (y in ' \n\r\t')] | ||||
|             while not hitEod: | ||||
|                 c = data[x] | ||||
|                 if len(retval) == 0 and c == "<" and data[x+1] == "~": | ||||
|                     x += 2 | ||||
|                     continue | ||||
|                 #elif c.isspace(): | ||||
|                 #    x += 1 | ||||
|                 #    continue | ||||
|                 elif c == 'z': | ||||
|                     assert len(group) == 0 | ||||
|                     retval += '\x00\x00\x00\x00' | ||||
|                     x += 1 | ||||
|                     continue | ||||
|                 elif c == "~" and data[x+1] == ">": | ||||
|                     if len(group) != 0: | ||||
|                         # cannot have a final group of just 1 char | ||||
|                         assert len(group) > 1 | ||||
|                         cnt = len(group) - 1 | ||||
|                         group += [ 85, 85, 85 ] | ||||
|                         hitEod = cnt | ||||
|                     else: | ||||
|                         break | ||||
|                 else: | ||||
|                     c = ord(c) - 33 | ||||
|                     assert c >= 0 and c < 85 | ||||
|                     group += [ c ] | ||||
|                 if len(group) >= 5: | ||||
|                     b = group[0] * (85**4) + \ | ||||
|                         group[1] * (85**3) + \ | ||||
|                         group[2] * (85**2) + \ | ||||
|                         group[3] * 85 + \ | ||||
|                         group[4] | ||||
|                     assert b < (2**32 - 1) | ||||
|                     c4 = chr((b >> 0) % 256) | ||||
|                     c3 = chr((b >> 8) % 256) | ||||
|                     c2 = chr((b >> 16) % 256) | ||||
|                     c1 = chr(b >> 24) | ||||
|                     retval += (c1 + c2 + c3 + c4) | ||||
|                     if hitEod: | ||||
|                         retval = retval[:-4+hitEod] | ||||
|                     group = [] | ||||
|                 x += 1 | ||||
|             return retval | ||||
|         else: | ||||
|             if isinstance(data, str): | ||||
|                 data = data.encode('ascii') | ||||
|             n = b = 0 | ||||
|             out = bytearray() | ||||
|             for c in data: | ||||
|                 if ord('!') <= c and c <= ord('u'): | ||||
|                     n += 1 | ||||
|                     b = b*85+(c-33) | ||||
|                     if n == 5: | ||||
|                         out += struct.pack(b'>L',b) | ||||
|                         n = b = 0 | ||||
|                 elif c == ord('z'): | ||||
|                     assert n == 0 | ||||
|                     out += b'\0\0\0\0' | ||||
|                 elif c == ord('~'): | ||||
|                     if n: | ||||
|                         for _ in range(5-n): | ||||
|                             b = b*85+84 | ||||
|                         out += struct.pack(b'>L',b)[:n-1] | ||||
|                     break | ||||
|             return bytes(out) | ||||
|     decode = staticmethod(decode) | ||||
| 
 | ||||
| 
 | ||||
| def decodeStreamData(stream): | ||||
|     from .generic import NameObject | ||||
|     filters = stream.get("/Filter", ()) | ||||
|     if len(filters) and not isinstance(filters[0], NameObject): | ||||
|         # we have a single filter instance | ||||
|         filters = (filters,) | ||||
|     data = stream._data | ||||
|     # If there is not data to decode we should not try to decode the data. | ||||
|     if data: | ||||
|         for filterType in filters: | ||||
|             if filterType == "/FlateDecode" or filterType == "/Fl": | ||||
|                 data = FlateDecode.decode(data, stream.get("/DecodeParms")) | ||||
|             elif filterType == "/ASCIIHexDecode" or filterType == "/AHx": | ||||
|                 data = ASCIIHexDecode.decode(data) | ||||
|             elif filterType == "/LZWDecode" or filterType == "/LZW": | ||||
|                 data = LZWDecode.decode(data, stream.get("/DecodeParms")) | ||||
|             elif filterType == "/ASCII85Decode" or filterType == "/A85": | ||||
|                 data = ASCII85Decode.decode(data) | ||||
|             elif filterType == "/Crypt": | ||||
|                 decodeParams = stream.get("/DecodeParams", {}) | ||||
|                 if "/Name" not in decodeParams and "/Type" not in decodeParams: | ||||
|                     pass | ||||
|                 else: | ||||
|                     raise NotImplementedError("/Crypt filter with /Name or /Type not supported yet") | ||||
|             else: | ||||
|                 # unsupported filter | ||||
|                 raise NotImplementedError("unsupported filter %s" % filterType) | ||||
|     return data | ||||
							
								
								
									
										1226
									
								
								vendor/PyPDF2/generic.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1226
									
								
								vendor/PyPDF2/generic.py
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										553
									
								
								vendor/PyPDF2/merger.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										553
									
								
								vendor/PyPDF2/merger.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,553 +0,0 @@ | |||
| # vim: sw=4:expandtab:foldmethod=marker | ||||
| # | ||||
| # Copyright (c) 2006, Mathieu Fenniak | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without | ||||
| # modification, are permitted provided that the following conditions are | ||||
| # met: | ||||
| # | ||||
| # * Redistributions of source code must retain the above copyright notice, | ||||
| # this list of conditions and the following disclaimer. | ||||
| # * Redistributions in binary form must reproduce the above copyright notice, | ||||
| # this list of conditions and the following disclaimer in the documentation | ||||
| # and/or other materials provided with the distribution. | ||||
| # * The name of the author may not be used to endorse or promote products | ||||
| # derived from this software without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||
| # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE | ||||
| # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | ||||
| # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | ||||
| # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||
| # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | ||||
| # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | ||||
| # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | ||||
| # POSSIBILITY OF SUCH DAMAGE. | ||||
| 
 | ||||
| from .generic import * | ||||
| from .utils import isString, str_ | ||||
| from .pdf import PdfFileReader, PdfFileWriter | ||||
| from .pagerange import PageRange | ||||
| from sys import version_info | ||||
| if version_info < ( 3, 0 ): | ||||
|     from cStringIO import StringIO | ||||
|     StreamIO = StringIO | ||||
| else: | ||||
|     from io import BytesIO | ||||
|     from io import FileIO as file | ||||
|     StreamIO = BytesIO | ||||
| 
 | ||||
| 
 | ||||
| class _MergedPage(object): | ||||
|     """ | ||||
|     _MergedPage is used internally by PdfFileMerger to collect necessary | ||||
|     information on each page that is being merged. | ||||
|     """ | ||||
|     def __init__(self, pagedata, src, id): | ||||
|         self.src = src | ||||
|         self.pagedata = pagedata | ||||
|         self.out_pagedata = None | ||||
|         self.id = id | ||||
| 
 | ||||
| 
 | ||||
| class PdfFileMerger(object): | ||||
|     """ | ||||
|     Initializes a PdfFileMerger object. PdfFileMerger merges multiple PDFs | ||||
|     into a single PDF. It can concatenate, slice, insert, or any combination | ||||
|     of the above. | ||||
| 
 | ||||
|     See the functions :meth:`merge()<merge>` (or :meth:`append()<append>`) | ||||
|     and :meth:`write()<write>` for usage information. | ||||
| 
 | ||||
|     :param bool strict: Determines whether user should be warned of all | ||||
|             problems and also causes some correctable problems to be fatal. | ||||
|             Defaults to ``True``. | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, strict=True): | ||||
|         self.inputs = [] | ||||
|         self.pages = [] | ||||
|         self.output = PdfFileWriter() | ||||
|         self.bookmarks = [] | ||||
|         self.named_dests = [] | ||||
|         self.id_count = 0 | ||||
|         self.strict = strict | ||||
| 
 | ||||
|     def merge(self, position, fileobj, bookmark=None, pages=None, import_bookmarks=True): | ||||
|         """ | ||||
|         Merges the pages from the given file into the output file at the | ||||
|         specified page number. | ||||
| 
 | ||||
|         :param int position: The *page number* to insert this file. File will | ||||
|             be inserted after the given number. | ||||
| 
 | ||||
|         :param fileobj: A File Object or an object that supports the standard read | ||||
|             and seek methods similar to a File Object. Could also be a | ||||
|             string representing a path to a PDF file. | ||||
| 
 | ||||
|         :param str bookmark: Optionally, you may specify a bookmark to be applied at | ||||
|             the beginning of the included file by supplying the text of the bookmark. | ||||
| 
 | ||||
|         :param pages: can be a :ref:`Page Range <page-range>` or a ``(start, stop[, step])`` tuple | ||||
|             to merge only the specified range of pages from the source | ||||
|             document into the output document. | ||||
| 
 | ||||
|         :param bool import_bookmarks: You may prevent the source document's bookmarks | ||||
|             from being imported by specifying this as ``False``. | ||||
|         """ | ||||
| 
 | ||||
|         # This parameter is passed to self.inputs.append and means | ||||
|         # that the stream used was created in this method. | ||||
|         my_file = False | ||||
| 
 | ||||
|         # If the fileobj parameter is a string, assume it is a path | ||||
|         # and create a file object at that location. If it is a file, | ||||
|         # copy the file's contents into a BytesIO (or StreamIO) stream object; if | ||||
|         # it is a PdfFileReader, copy that reader's stream into a | ||||
|         # BytesIO (or StreamIO) stream. | ||||
|         # If fileobj is none of the above types, it is not modified | ||||
|         decryption_key = None | ||||
|         if isString(fileobj): | ||||
|             fileobj = file(fileobj, 'rb') | ||||
|             my_file = True | ||||
|         elif isinstance(fileobj, file): | ||||
|             fileobj.seek(0) | ||||
|             filecontent = fileobj.read() | ||||
|             fileobj = StreamIO(filecontent) | ||||
|             my_file = True | ||||
|         elif isinstance(fileobj, PdfFileReader): | ||||
|             orig_tell = fileobj.stream.tell() | ||||
|             fileobj.stream.seek(0) | ||||
|             filecontent = StreamIO(fileobj.stream.read()) | ||||
|             fileobj.stream.seek(orig_tell) # reset the stream to its original location | ||||
|             fileobj = filecontent | ||||
|             if hasattr(fileobj, '_decryption_key'): | ||||
|                 decryption_key = fileobj._decryption_key | ||||
|             my_file = True | ||||
| 
 | ||||
|         # Create a new PdfFileReader instance using the stream | ||||
|         # (either file or BytesIO or StringIO) created above | ||||
|         pdfr = PdfFileReader(fileobj, strict=self.strict) | ||||
|         if decryption_key is not None: | ||||
|             pdfr._decryption_key = decryption_key | ||||
| 
 | ||||
|         # Find the range of pages to merge. | ||||
|         if pages == None: | ||||
|             pages = (0, pdfr.getNumPages()) | ||||
|         elif isinstance(pages, PageRange): | ||||
|             pages = pages.indices(pdfr.getNumPages()) | ||||
|         elif not isinstance(pages, tuple): | ||||
|             raise TypeError('"pages" must be a tuple of (start, stop[, step])') | ||||
| 
 | ||||
|         srcpages = [] | ||||
|         if bookmark: | ||||
|             bookmark = Bookmark(TextStringObject(bookmark), NumberObject(self.id_count), NameObject('/Fit')) | ||||
| 
 | ||||
|         outline = [] | ||||
|         if import_bookmarks: | ||||
|             outline = pdfr.getOutlines() | ||||
|             outline = self._trim_outline(pdfr, outline, pages) | ||||
| 
 | ||||
|         if bookmark: | ||||
|             self.bookmarks += [bookmark, outline] | ||||
|         else: | ||||
|             self.bookmarks += outline | ||||
| 
 | ||||
|         dests = pdfr.namedDestinations | ||||
|         dests = self._trim_dests(pdfr, dests, pages) | ||||
|         self.named_dests += dests | ||||
| 
 | ||||
|         # Gather all the pages that are going to be merged | ||||
|         for i in range(*pages): | ||||
|             pg = pdfr.getPage(i) | ||||
| 
 | ||||
|             id = self.id_count | ||||
|             self.id_count += 1 | ||||
| 
 | ||||
|             mp = _MergedPage(pg, pdfr, id) | ||||
| 
 | ||||
|             srcpages.append(mp) | ||||
| 
 | ||||
|         self._associate_dests_to_pages(srcpages) | ||||
|         self._associate_bookmarks_to_pages(srcpages) | ||||
| 
 | ||||
|         # Slice to insert the pages at the specified position | ||||
|         self.pages[position:position] = srcpages | ||||
| 
 | ||||
|         # Keep track of our input files so we can close them later | ||||
|         self.inputs.append((fileobj, pdfr, my_file)) | ||||
| 
 | ||||
|     def append(self, fileobj, bookmark=None, pages=None, import_bookmarks=True): | ||||
|         """ | ||||
|         Identical to the :meth:`merge()<merge>` method, but assumes you want to concatenate | ||||
|         all pages onto the end of the file instead of specifying a position. | ||||
| 
 | ||||
|         :param fileobj: A File Object or an object that supports the standard read | ||||
|             and seek methods similar to a File Object. Could also be a | ||||
|             string representing a path to a PDF file. | ||||
| 
 | ||||
|         :param str bookmark: Optionally, you may specify a bookmark to be applied at | ||||
|             the beginning of the included file by supplying the text of the bookmark. | ||||
| 
 | ||||
|         :param pages: can be a :ref:`Page Range <page-range>` or a ``(start, stop[, step])`` tuple | ||||
|             to merge only the specified range of pages from the source | ||||
|             document into the output document. | ||||
| 
 | ||||
|         :param bool import_bookmarks: You may prevent the source document's bookmarks | ||||
|             from being imported by specifying this as ``False``. | ||||
|         """ | ||||
| 
 | ||||
|         self.merge(len(self.pages), fileobj, bookmark, pages, import_bookmarks) | ||||
| 
 | ||||
|     def write(self, fileobj): | ||||
|         """ | ||||
|         Writes all data that has been merged to the given output file. | ||||
| 
 | ||||
|         :param fileobj: Output file. Can be a filename or any kind of | ||||
|             file-like object. | ||||
|         """ | ||||
|         my_file = False | ||||
|         if isString(fileobj): | ||||
|             fileobj = file(fileobj, 'wb') | ||||
|             my_file = True | ||||
| 
 | ||||
|         # Add pages to the PdfFileWriter | ||||
|         # The commented out line below was replaced with the two lines below it to allow PdfFileMerger to work with PyPdf 1.13 | ||||
|         for page in self.pages: | ||||
|             self.output.addPage(page.pagedata) | ||||
|             page.out_pagedata = self.output.getReference(self.output._pages.getObject()["/Kids"][-1].getObject()) | ||||
|             #idnum = self.output._objects.index(self.output._pages.getObject()["/Kids"][-1].getObject()) + 1 | ||||
|             #page.out_pagedata = IndirectObject(idnum, 0, self.output) | ||||
| 
 | ||||
|         # Once all pages are added, create bookmarks to point at those pages | ||||
|         self._write_dests() | ||||
|         self._write_bookmarks() | ||||
| 
 | ||||
|         # Write the output to the file | ||||
|         self.output.write(fileobj) | ||||
| 
 | ||||
|         if my_file: | ||||
|             fileobj.close() | ||||
| 
 | ||||
|     def close(self): | ||||
|         """ | ||||
|         Shuts all file descriptors (input and output) and clears all memory | ||||
|         usage. | ||||
|         """ | ||||
|         self.pages = [] | ||||
|         for fo, pdfr, mine in self.inputs: | ||||
|             if mine: | ||||
|                 fo.close() | ||||
| 
 | ||||
|         self.inputs = [] | ||||
|         self.output = None | ||||
| 
 | ||||
|     def addMetadata(self, infos): | ||||
|         """ | ||||
|         Add custom metadata to the output. | ||||
| 
 | ||||
|         :param dict infos: a Python dictionary where each key is a field | ||||
|             and each value is your new metadata. | ||||
|             Example: ``{u'/Title': u'My title'}`` | ||||
|         """ | ||||
|         self.output.addMetadata(infos) | ||||
| 
 | ||||
|     def setPageLayout(self, layout): | ||||
|         """ | ||||
|         Set the page layout | ||||
| 
 | ||||
|         :param str layout: The page layout to be used | ||||
| 
 | ||||
|         Valid layouts are: | ||||
|              /NoLayout        Layout explicitly not specified | ||||
|              /SinglePage      Show one page at a time | ||||
|              /OneColumn       Show one column at a time | ||||
|              /TwoColumnLeft   Show pages in two columns, odd-numbered pages on the left | ||||
|              /TwoColumnRight  Show pages in two columns, odd-numbered pages on the right | ||||
|              /TwoPageLeft     Show two pages at a time, odd-numbered pages on the left | ||||
|              /TwoPageRight    Show two pages at a time, odd-numbered pages on the right | ||||
|         """ | ||||
|         self.output.setPageLayout(layout) | ||||
| 
 | ||||
|     def setPageMode(self, mode): | ||||
|         """ | ||||
|         Set the page mode. | ||||
| 
 | ||||
|         :param str mode: The page mode to use. | ||||
| 
 | ||||
|         Valid modes are: | ||||
|             /UseNone         Do not show outlines or thumbnails panels | ||||
|             /UseOutlines     Show outlines (aka bookmarks) panel | ||||
|             /UseThumbs       Show page thumbnails panel | ||||
|             /FullScreen      Fullscreen view | ||||
|             /UseOC           Show Optional Content Group (OCG) panel | ||||
|             /UseAttachments  Show attachments panel | ||||
|         """ | ||||
|         self.output.setPageMode(mode) | ||||
| 
 | ||||
|     def _trim_dests(self, pdf, dests, pages): | ||||
|         """ | ||||
|         Removes any named destinations that are not a part of the specified | ||||
|         page set. | ||||
|         """ | ||||
|         new_dests = [] | ||||
|         prev_header_added = True | ||||
|         for k, o in list(dests.items()): | ||||
|             for j in range(*pages): | ||||
|                 if pdf.getPage(j).getObject() == o['/Page'].getObject(): | ||||
|                     o[NameObject('/Page')] = o['/Page'].getObject() | ||||
|                     assert str_(k) == str_(o['/Title']) | ||||
|                     new_dests.append(o) | ||||
|                     break | ||||
|         return new_dests | ||||
| 
 | ||||
|     def _trim_outline(self, pdf, outline, pages): | ||||
|         """ | ||||
|         Removes any outline/bookmark entries that are not a part of the | ||||
|         specified page set. | ||||
|         """ | ||||
|         new_outline = [] | ||||
|         prev_header_added = True | ||||
|         for i, o in enumerate(outline): | ||||
|             if isinstance(o, list): | ||||
|                 sub = self._trim_outline(pdf, o, pages) | ||||
|                 if sub: | ||||
|                     if not prev_header_added: | ||||
|                         new_outline.append(outline[i-1]) | ||||
|                     new_outline.append(sub) | ||||
|             else: | ||||
|                 prev_header_added = False | ||||
|                 for j in range(*pages): | ||||
|                     if pdf.getPage(j).getObject() == o['/Page'].getObject(): | ||||
|                         o[NameObject('/Page')] = o['/Page'].getObject() | ||||
|                         new_outline.append(o) | ||||
|                         prev_header_added = True | ||||
|                         break | ||||
|         return new_outline | ||||
| 
 | ||||
|     def _write_dests(self): | ||||
|         dests = self.named_dests | ||||
| 
 | ||||
|         for v in dests: | ||||
|             pageno = None | ||||
|             pdf = None | ||||
|             if '/Page' in v: | ||||
|                 for i, p in enumerate(self.pages): | ||||
|                     if p.id == v['/Page']: | ||||
|                         v[NameObject('/Page')] = p.out_pagedata | ||||
|                         pageno = i | ||||
|                         pdf = p.src | ||||
|                         break | ||||
|             if pageno != None: | ||||
|                 self.output.addNamedDestinationObject(v) | ||||
| 
 | ||||
|     def _write_bookmarks(self, bookmarks=None, parent=None): | ||||
| 
 | ||||
|         if bookmarks == None: | ||||
|             bookmarks = self.bookmarks | ||||
| 
 | ||||
|         last_added = None | ||||
|         for b in bookmarks: | ||||
|             if isinstance(b, list): | ||||
|                 self._write_bookmarks(b, last_added) | ||||
|                 continue | ||||
| 
 | ||||
|             pageno = None | ||||
|             pdf = None | ||||
|             if '/Page' in b: | ||||
|                 for i, p in enumerate(self.pages): | ||||
|                     if p.id == b['/Page']: | ||||
|                         #b[NameObject('/Page')] = p.out_pagedata | ||||
|                         args = [NumberObject(p.id), NameObject(b['/Type'])] | ||||
|                         #nothing more to add | ||||
|                         #if b['/Type'] == '/Fit' or b['/Type'] == '/FitB' | ||||
|                         if b['/Type'] == '/FitH' or b['/Type'] == '/FitBH': | ||||
|                             if '/Top' in b and not isinstance(b['/Top'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Top'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             del b['/Top'] | ||||
|                         elif b['/Type'] == '/FitV' or b['/Type'] == '/FitBV': | ||||
|                             if '/Left' in b and not isinstance(b['/Left'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Left'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             del b['/Left'] | ||||
|                         elif b['/Type'] == '/XYZ': | ||||
|                             if '/Left' in b and not isinstance(b['/Left'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Left'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             if '/Top' in b and not isinstance(b['/Top'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Top'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             if '/Zoom' in b and not isinstance(b['/Zoom'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Zoom'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             del b['/Top'], b['/Zoom'], b['/Left'] | ||||
|                         elif b['/Type'] == '/FitR': | ||||
|                             if '/Left' in b and not isinstance(b['/Left'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Left'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             if '/Bottom' in b and not isinstance(b['/Bottom'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Bottom'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             if '/Right' in b and not isinstance(b['/Right'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Right'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             if '/Top' in b and not isinstance(b['/Top'], NullObject): | ||||
|                                 args.append(FloatObject(b['/Top'])) | ||||
|                             else: | ||||
|                                 args.append(FloatObject(0)) | ||||
|                             del b['/Left'], b['/Right'], b['/Bottom'], b['/Top'] | ||||
| 
 | ||||
|                         b[NameObject('/A')] = DictionaryObject({NameObject('/S'): NameObject('/GoTo'), NameObject('/D'): ArrayObject(args)}) | ||||
| 
 | ||||
|                         pageno = i | ||||
|                         pdf = p.src | ||||
|                         break | ||||
|             if pageno != None: | ||||
|                 del b['/Page'], b['/Type'] | ||||
|                 last_added = self.output.addBookmarkDict(b, parent) | ||||
| 
 | ||||
|     def _associate_dests_to_pages(self, pages): | ||||
|         for nd in self.named_dests: | ||||
|             pageno = None | ||||
|             np = nd['/Page'] | ||||
| 
 | ||||
|             if isinstance(np, NumberObject): | ||||
|                 continue | ||||
| 
 | ||||
|             for p in pages: | ||||
|                 if np.getObject() == p.pagedata.getObject(): | ||||
|                     pageno = p.id | ||||
| 
 | ||||
|             if pageno != None: | ||||
|                 nd[NameObject('/Page')] = NumberObject(pageno) | ||||
|             else: | ||||
|                 raise ValueError("Unresolved named destination '%s'" % (nd['/Title'],)) | ||||
| 
 | ||||
|     def _associate_bookmarks_to_pages(self, pages, bookmarks=None): | ||||
|         if bookmarks == None: | ||||
|             bookmarks = self.bookmarks | ||||
| 
 | ||||
|         for b in bookmarks: | ||||
|             if isinstance(b, list): | ||||
|                 self._associate_bookmarks_to_pages(pages, b) | ||||
|                 continue | ||||
| 
 | ||||
|             pageno = None | ||||
|             bp = b['/Page'] | ||||
| 
 | ||||
|             if isinstance(bp, NumberObject): | ||||
|                 continue | ||||
| 
 | ||||
|             for p in pages: | ||||
|                 if bp.getObject() == p.pagedata.getObject(): | ||||
|                     pageno = p.id | ||||
| 
 | ||||
|             if pageno != None: | ||||
|                 b[NameObject('/Page')] = NumberObject(pageno) | ||||
|             else: | ||||
|                 raise ValueError("Unresolved bookmark '%s'" % (b['/Title'],)) | ||||
| 
 | ||||
|     def findBookmark(self, bookmark, root=None): | ||||
|         if root == None: | ||||
|             root = self.bookmarks | ||||
| 
 | ||||
|         for i, b in enumerate(root): | ||||
|             if isinstance(b, list): | ||||
|                 res = self.findBookmark(bookmark, b) | ||||
|                 if res: | ||||
|                     return [i] + res | ||||
|             elif b == bookmark or b['/Title'] == bookmark: | ||||
|                 return [i] | ||||
| 
 | ||||
|         return None | ||||
| 
 | ||||
|     def addBookmark(self, title, pagenum, parent=None): | ||||
|         """ | ||||
|         Add a bookmark to this PDF file. | ||||
| 
 | ||||
|         :param str title: Title to use for this bookmark. | ||||
|         :param int pagenum: Page number this bookmark will point to. | ||||
|         :param parent: A reference to a parent bookmark to create nested | ||||
|             bookmarks. | ||||
|         """ | ||||
|         if parent == None: | ||||
|             iloc = [len(self.bookmarks)-1] | ||||
|         elif isinstance(parent, list): | ||||
|             iloc = parent | ||||
|         else: | ||||
|             iloc = self.findBookmark(parent) | ||||
| 
 | ||||
|         dest = Bookmark(TextStringObject(title), NumberObject(pagenum), NameObject('/FitH'), NumberObject(826)) | ||||
| 
 | ||||
|         if parent == None: | ||||
|             self.bookmarks.append(dest) | ||||
|         else: | ||||
|             bmparent = self.bookmarks | ||||
|             for i in iloc[:-1]: | ||||
|                 bmparent = bmparent[i] | ||||
|             npos = iloc[-1]+1 | ||||
|             if npos < len(bmparent) and isinstance(bmparent[npos], list): | ||||
|                 bmparent[npos].append(dest) | ||||
|             else: | ||||
|                 bmparent.insert(npos, [dest]) | ||||
|         return dest | ||||
| 
 | ||||
|     def addNamedDestination(self, title, pagenum): | ||||
|         """ | ||||
|         Add a destination to the output. | ||||
| 
 | ||||
|         :param str title: Title to use | ||||
|         :param int pagenum: Page number this destination points at. | ||||
|         """ | ||||
| 
 | ||||
|         dest = Destination(TextStringObject(title), NumberObject(pagenum), NameObject('/FitH'), NumberObject(826)) | ||||
|         self.named_dests.append(dest) | ||||
| 
 | ||||
| 
 | ||||
| class OutlinesObject(list): | ||||
|     def __init__(self, pdf, tree, parent=None): | ||||
|         list.__init__(self) | ||||
|         self.tree = tree | ||||
|         self.pdf = pdf | ||||
|         self.parent = parent | ||||
| 
 | ||||
|     def remove(self, index): | ||||
|         obj = self[index] | ||||
|         del self[index] | ||||
|         self.tree.removeChild(obj) | ||||
| 
 | ||||
|     def add(self, title, pagenum): | ||||
|         pageRef = self.pdf.getObject(self.pdf._pages)['/Kids'][pagenum] | ||||
|         action = DictionaryObject() | ||||
|         action.update({ | ||||
|             NameObject('/D') : ArrayObject([pageRef, NameObject('/FitH'), NumberObject(826)]), | ||||
|             NameObject('/S') : NameObject('/GoTo') | ||||
|         }) | ||||
|         actionRef = self.pdf._addObject(action) | ||||
|         bookmark = TreeObject() | ||||
| 
 | ||||
|         bookmark.update({ | ||||
|             NameObject('/A'): actionRef, | ||||
|             NameObject('/Title'): createStringObject(title), | ||||
|         }) | ||||
| 
 | ||||
|         self.pdf._addObject(bookmark) | ||||
| 
 | ||||
|         self.tree.addChild(bookmark) | ||||
| 
 | ||||
|     def removeAll(self): | ||||
|         for child in [x for x in self.tree.children()]: | ||||
|             self.tree.removeChild(child) | ||||
|             self.pop() | ||||
							
								
								
									
										152
									
								
								vendor/PyPDF2/pagerange.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										152
									
								
								vendor/PyPDF2/pagerange.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,152 +0,0 @@ | |||
| #!/usr/bin/env python | ||||
| """ | ||||
| Representation and utils for ranges of PDF file pages. | ||||
| 
 | ||||
| Copyright (c) 2014, Steve Witham <switham_github@mac-guyver.com>. | ||||
| All rights reserved. This software is available under a BSD license; | ||||
| see https://github.com/mstamy2/PyPDF2/blob/master/LICENSE | ||||
| """ | ||||
| 
 | ||||
| import re | ||||
| from .utils import isString | ||||
| 
 | ||||
| _INT_RE = r"(0|-?[1-9]\d*)"  # A decimal int, don't allow "-0". | ||||
| PAGE_RANGE_RE = "^({int}|({int}?(:{int}?(:{int}?)?)))$".format(int=_INT_RE) | ||||
| # groups:         12     34     5 6     7 8 | ||||
| 
 | ||||
| 
 | ||||
| class ParseError(Exception): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| PAGE_RANGE_HELP = """Remember, page indices start with zero. | ||||
|         Page range expression examples: | ||||
|             :     all pages.                   -1    last page. | ||||
|             22    just the 23rd page.          :-1   all but the last page. | ||||
|             0:3   the first three pages.       -2    second-to-last page. | ||||
|             :3    the first three pages.       -2:   last two pages. | ||||
|             5:    from the sixth page onward.  -3:-1 third & second to last. | ||||
|         The third, "stride" or "step" number is also recognized. | ||||
|             ::2       0 2 4 ... to the end.    3:0:-1    3 2 1 but not 0. | ||||
|             1:10:2    1 3 5 7 9                2::-1     2 1 0. | ||||
|             ::-1      all pages in reverse order. | ||||
| """ | ||||
| 
 | ||||
| 
 | ||||
| class PageRange(object): | ||||
|     """ | ||||
|     A slice-like representation of a range of page indices, | ||||
|         i.e. page numbers, only starting at zero. | ||||
|     The syntax is like what you would put between brackets [ ]. | ||||
|     The slice is one of the few Python types that can't be subclassed, | ||||
|     but this class converts to and from slices, and allows similar use. | ||||
|       o  PageRange(str) parses a string representing a page range. | ||||
|       o  PageRange(slice) directly "imports" a slice. | ||||
|       o  to_slice() gives the equivalent slice. | ||||
|       o  str() and repr() allow printing. | ||||
|       o  indices(n) is like slice.indices(n). | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, arg): | ||||
|         """ | ||||
|         Initialize with either a slice -- giving the equivalent page range, | ||||
|         or a PageRange object -- making a copy, | ||||
|         or a string like | ||||
|             "int", "[int]:[int]" or "[int]:[int]:[int]", | ||||
|             where the brackets indicate optional ints. | ||||
|         {page_range_help} | ||||
|         Note the difference between this notation and arguments to slice(): | ||||
|             slice(3) means the first three pages; | ||||
|             PageRange("3") means the range of only the fourth page. | ||||
|             However PageRange(slice(3)) means the first three pages. | ||||
|         """ | ||||
|         if isinstance(arg, slice): | ||||
|             self._slice = arg | ||||
|             return | ||||
| 
 | ||||
|         if isinstance(arg, PageRange): | ||||
|             self._slice = arg.to_slice() | ||||
|             return | ||||
| 
 | ||||
|         m = isString(arg) and re.match(PAGE_RANGE_RE, arg) | ||||
|         if not m: | ||||
|             raise ParseError(arg) | ||||
|         elif m.group(2): | ||||
|             # Special case: just an int means a range of one page. | ||||
|             start = int(m.group(2)) | ||||
|             stop = start + 1 if start != -1 else None | ||||
|             self._slice = slice(start, stop) | ||||
|         else: | ||||
|             self._slice = slice(*[int(g) if g else None | ||||
|                                   for g in m.group(4, 6, 8)]) | ||||
| 
 | ||||
|     # Just formatting this when there is __doc__ for __init__ | ||||
|     if __init__.__doc__: | ||||
|         __init__.__doc__ = __init__.__doc__.format(page_range_help=PAGE_RANGE_HELP) | ||||
| 
 | ||||
|     @staticmethod | ||||
|     def valid(input): | ||||
|         """ True if input is a valid initializer for a PageRange. """ | ||||
|         return isinstance(input, slice)  or \ | ||||
|                isinstance(input, PageRange) or \ | ||||
|                (isString(input) | ||||
|                 and bool(re.match(PAGE_RANGE_RE, input))) | ||||
| 
 | ||||
|     def to_slice(self): | ||||
|         """ Return the slice equivalent of this page range. """ | ||||
|         return self._slice | ||||
| 
 | ||||
|     def __str__(self): | ||||
|         """ A string like "1:2:3". """ | ||||
|         s = self._slice | ||||
|         if s.step == None: | ||||
|             if s.start != None  and  s.stop == s.start + 1: | ||||
|                 return str(s.start) | ||||
| 
 | ||||
|             indices = s.start, s.stop | ||||
|         else: | ||||
|             indices = s.start, s.stop, s.step | ||||
|         return ':'.join("" if i == None else str(i) for i in indices) | ||||
| 
 | ||||
|     def __repr__(self): | ||||
|         """ A string like "PageRange('1:2:3')". """ | ||||
|         return "PageRange(" + repr(str(self)) + ")" | ||||
| 
 | ||||
|     def indices(self, n): | ||||
|         """ | ||||
|         n is the length of the list of pages to choose from. | ||||
|         Returns arguments for range().  See help(slice.indices). | ||||
|         """ | ||||
|         return self._slice.indices(n) | ||||
| 
 | ||||
| 
 | ||||
| PAGE_RANGE_ALL = PageRange(":")  # The range of all pages. | ||||
| 
 | ||||
| 
 | ||||
| def parse_filename_page_ranges(args): | ||||
|     """ | ||||
|     Given a list of filenames and page ranges, return a list of | ||||
|     (filename, page_range) pairs. | ||||
|     First arg must be a filename; other ags are filenames, page-range | ||||
|     expressions, slice objects, or PageRange objects. | ||||
|     A filename not followed by a page range indicates all pages of the file. | ||||
|     """ | ||||
|     pairs = [] | ||||
|     pdf_filename = None | ||||
|     did_page_range = False | ||||
|     for arg in args + [None]: | ||||
|         if PageRange.valid(arg): | ||||
|             if not pdf_filename: | ||||
|                 raise ValueError("The first argument must be a filename, " \ | ||||
|                                  "not a page range.") | ||||
| 
 | ||||
|             pairs.append( (pdf_filename, PageRange(arg)) ) | ||||
|             did_page_range = True | ||||
|         else: | ||||
|             # New filename or end of list--do all of the previous file? | ||||
|             if pdf_filename and not did_page_range: | ||||
|                 pairs.append( (pdf_filename, PAGE_RANGE_ALL) ) | ||||
| 
 | ||||
|             pdf_filename = arg | ||||
|             did_page_range = False | ||||
|     return pairs | ||||
							
								
								
									
										3004
									
								
								vendor/PyPDF2/pdf.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3004
									
								
								vendor/PyPDF2/pdf.py
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										295
									
								
								vendor/PyPDF2/utils.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										295
									
								
								vendor/PyPDF2/utils.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,295 +0,0 @@ | |||
| # Copyright (c) 2006, Mathieu Fenniak | ||||
| # All rights reserved. | ||||
| # | ||||
| # Redistribution and use in source and binary forms, with or without | ||||
| # modification, are permitted provided that the following conditions are | ||||
| # met: | ||||
| # | ||||
| # * Redistributions of source code must retain the above copyright notice, | ||||
| # this list of conditions and the following disclaimer. | ||||
| # * Redistributions in binary form must reproduce the above copyright notice, | ||||
| # this list of conditions and the following disclaimer in the documentation | ||||
| # and/or other materials provided with the distribution. | ||||
| # * The name of the author may not be used to endorse or promote products | ||||
| # derived from this software without specific prior written permission. | ||||
| # | ||||
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||
| # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||
| # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||
| # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE | ||||
| # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | ||||
| # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | ||||
| # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||
| # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | ||||
| # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | ||||
| # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | ||||
| # POSSIBILITY OF SUCH DAMAGE. | ||||
| 
 | ||||
| """ | ||||
| Utility functions for PDF library. | ||||
| """ | ||||
| __author__ = "Mathieu Fenniak" | ||||
| __author_email__ = "biziqe@mathieu.fenniak.net" | ||||
| 
 | ||||
| 
 | ||||
| import sys | ||||
| 
 | ||||
| try: | ||||
|     import __builtin__ as builtins | ||||
| except ImportError:  # Py3 | ||||
|     import builtins | ||||
| 
 | ||||
| 
 | ||||
| xrange_fn = getattr(builtins, "xrange", range) | ||||
| _basestring = getattr(builtins, "basestring", str) | ||||
| 
 | ||||
| bytes_type = type(bytes()) # Works the same in Python 2.X and 3.X | ||||
| string_type = getattr(builtins, "unicode", str) | ||||
| int_types = (int, long) if sys.version_info[0] < 3 else (int,) | ||||
| 
 | ||||
| 
 | ||||
| # Make basic type tests more consistent | ||||
| def isString(s): | ||||
|     """Test if arg is a string. Compatible with Python 2 and 3.""" | ||||
|     return isinstance(s, _basestring) | ||||
| 
 | ||||
| 
 | ||||
| def isInt(n): | ||||
|     """Test if arg is an int. Compatible with Python 2 and 3.""" | ||||
|     return isinstance(n, int_types) | ||||
| 
 | ||||
| 
 | ||||
| def isBytes(b): | ||||
|     """Test if arg is a bytes instance. Compatible with Python 2 and 3.""" | ||||
|     return isinstance(b, bytes_type) | ||||
| 
 | ||||
| 
 | ||||
| #custom implementation of warnings.formatwarning | ||||
| def formatWarning(message, category, filename, lineno, line=None): | ||||
|     file = filename.replace("/", "\\").rsplit("\\", 1)[1] # find the file name | ||||
|     return "%s: %s [%s:%s]\n" % (category.__name__, message, file, lineno) | ||||
| 
 | ||||
| 
 | ||||
| def readUntilWhitespace(stream, maxchars=None): | ||||
|     """ | ||||
|     Reads non-whitespace characters and returns them. | ||||
|     Stops upon encountering whitespace or when maxchars is reached. | ||||
|     """ | ||||
|     txt = b_("") | ||||
|     while True: | ||||
|         tok = stream.read(1) | ||||
|         if tok.isspace() or not tok: | ||||
|             break | ||||
|         txt += tok | ||||
|         if len(txt) == maxchars: | ||||
|             break | ||||
|     return txt | ||||
| 
 | ||||
| 
 | ||||
| def readNonWhitespace(stream): | ||||
|     """ | ||||
|     Finds and reads the next non-whitespace character (ignores whitespace). | ||||
|     """ | ||||
|     tok = WHITESPACES[0] | ||||
|     while tok in WHITESPACES: | ||||
|         tok = stream.read(1) | ||||
|     return tok | ||||
| 
 | ||||
| 
 | ||||
| def skipOverWhitespace(stream): | ||||
|     """ | ||||
|     Similar to readNonWhitespace, but returns a Boolean if more than | ||||
|     one whitespace character was read. | ||||
|     """ | ||||
|     tok = WHITESPACES[0] | ||||
|     cnt = 0; | ||||
|     while tok in WHITESPACES: | ||||
|         tok = stream.read(1) | ||||
|         cnt+=1 | ||||
|     return (cnt > 1) | ||||
| 
 | ||||
| 
 | ||||
| def skipOverComment(stream): | ||||
|     tok = stream.read(1) | ||||
|     stream.seek(-1, 1) | ||||
|     if tok == b_('%'): | ||||
|         while tok not in (b_('\n'), b_('\r')): | ||||
|             tok = stream.read(1) | ||||
| 
 | ||||
| 
 | ||||
| def readUntilRegex(stream, regex, ignore_eof=False): | ||||
|     """ | ||||
|     Reads until the regular expression pattern matched (ignore the match) | ||||
|     Raise PdfStreamError on premature end-of-file. | ||||
|     :param bool ignore_eof: If true, ignore end-of-line and return immediately | ||||
|     """ | ||||
|     name = b_('') | ||||
|     while True: | ||||
|         tok = stream.read(16) | ||||
|         if not tok: | ||||
|             # stream has truncated prematurely | ||||
|             if ignore_eof == True: | ||||
|                 return name | ||||
|             else: | ||||
|                 raise PdfStreamError("Stream has ended unexpectedly") | ||||
|         m = regex.search(tok) | ||||
|         if m is not None: | ||||
|             name += tok[:m.start()] | ||||
|             stream.seek(m.start()-len(tok), 1) | ||||
|             break | ||||
|         name += tok | ||||
|     return name | ||||
| 
 | ||||
| 
 | ||||
| class ConvertFunctionsToVirtualList(object): | ||||
|     def __init__(self, lengthFunction, getFunction): | ||||
|         self.lengthFunction = lengthFunction | ||||
|         self.getFunction = getFunction | ||||
| 
 | ||||
|     def __len__(self): | ||||
|         return self.lengthFunction() | ||||
| 
 | ||||
|     def __getitem__(self, index): | ||||
|         if isinstance(index, slice): | ||||
|             indices = xrange_fn(*index.indices(len(self))) | ||||
|             cls = type(self) | ||||
|             return cls(indices.__len__, lambda idx: self[indices[idx]]) | ||||
|         if not isInt(index): | ||||
|             raise TypeError("sequence indices must be integers") | ||||
|         len_self = len(self) | ||||
|         if index < 0: | ||||
|             # support negative indexes | ||||
|             index = len_self + index | ||||
|         if index < 0 or index >= len_self: | ||||
|             raise IndexError("sequence index out of range") | ||||
|         return self.getFunction(index) | ||||
| 
 | ||||
| 
 | ||||
| def RC4_encrypt(key, plaintext): | ||||
|     S = [i for i in range(256)] | ||||
|     j = 0 | ||||
|     for i in range(256): | ||||
|         j = (j + S[i] + ord_(key[i % len(key)])) % 256 | ||||
|         S[i], S[j] = S[j], S[i] | ||||
|     i, j = 0, 0 | ||||
|     retval = b_("") | ||||
|     for x in range(len(plaintext)): | ||||
|         i = (i + 1) % 256 | ||||
|         j = (j + S[i]) % 256 | ||||
|         S[i], S[j] = S[j], S[i] | ||||
|         t = S[(S[i] + S[j]) % 256] | ||||
|         retval += b_(chr(ord_(plaintext[x]) ^ t)) | ||||
|     return retval | ||||
| 
 | ||||
| 
 | ||||
| def matrixMultiply(a, b): | ||||
|     return [[sum([float(i)*float(j) | ||||
|                   for i, j in zip(row, col)] | ||||
|                 ) for col in zip(*b)] | ||||
|             for row in a] | ||||
| 
 | ||||
| 
 | ||||
| def markLocation(stream): | ||||
|     """Creates text file showing current location in context.""" | ||||
|     # Mainly for debugging | ||||
|     RADIUS = 5000 | ||||
|     stream.seek(-RADIUS, 1) | ||||
|     outputDoc = open('PyPDF2_pdfLocation.txt', 'w') | ||||
|     outputDoc.write(stream.read(RADIUS)) | ||||
|     outputDoc.write('HERE') | ||||
|     outputDoc.write(stream.read(RADIUS)) | ||||
|     outputDoc.close() | ||||
|     stream.seek(-RADIUS, 1) | ||||
| 
 | ||||
| 
 | ||||
| class PyPdfError(Exception): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class PdfReadError(PyPdfError): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class PageSizeNotDefinedError(PyPdfError): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class PdfReadWarning(UserWarning): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class PdfStreamError(PdfReadError): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| if sys.version_info[0] < 3: | ||||
|     def b_(s): | ||||
|         return s | ||||
| else: | ||||
|     B_CACHE = {} | ||||
| 
 | ||||
|     def b_(s): | ||||
|         bc = B_CACHE | ||||
|         if s in bc: | ||||
|             return bc[s] | ||||
|         if type(s) == bytes: | ||||
|             return s | ||||
|         else: | ||||
|             r = s.encode('latin-1') | ||||
|             if len(s) < 2: | ||||
|                 bc[s] = r | ||||
|             return r | ||||
| 
 | ||||
| 
 | ||||
| def u_(s): | ||||
|     if sys.version_info[0] < 3: | ||||
|         return unicode(s, 'unicode_escape') | ||||
|     else: | ||||
|         return s | ||||
| 
 | ||||
| 
 | ||||
| def str_(b): | ||||
|     if sys.version_info[0] < 3: | ||||
|         return b | ||||
|     else: | ||||
|         if type(b) == bytes: | ||||
|             return b.decode('latin-1') | ||||
|         else: | ||||
|             return b | ||||
| 
 | ||||
| 
 | ||||
| def ord_(b): | ||||
|     if sys.version_info[0] < 3 or type(b) == str: | ||||
|         return ord(b) | ||||
|     else: | ||||
|         return b | ||||
| 
 | ||||
| 
 | ||||
| def chr_(c): | ||||
|     if sys.version_info[0] < 3: | ||||
|         return c | ||||
|     else: | ||||
|         return chr(c) | ||||
| 
 | ||||
| 
 | ||||
| def barray(b): | ||||
|     if sys.version_info[0] < 3: | ||||
|         return b | ||||
|     else: | ||||
|         return bytearray(b) | ||||
| 
 | ||||
| 
 | ||||
| def hexencode(b): | ||||
|     if sys.version_info[0] < 3: | ||||
|         return b.encode('hex') | ||||
|     else: | ||||
|         import codecs | ||||
|         coder = codecs.getencoder('hex_codec') | ||||
|         return coder(b)[0] | ||||
| 
 | ||||
| 
 | ||||
| def hexStr(num): | ||||
|     return hex(num).replace('L', '') | ||||
| 
 | ||||
| 
 | ||||
| WHITESPACES = [b_(x) for x in [' ', '\n', '\r', '\t', '\x00']] | ||||
							
								
								
									
										358
									
								
								vendor/PyPDF2/xmp.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										358
									
								
								vendor/PyPDF2/xmp.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,358 +0,0 @@ | |||
| import re | ||||
| import datetime | ||||
| import decimal | ||||
| from .generic import PdfObject | ||||
| from xml.dom import getDOMImplementation | ||||
| from xml.dom.minidom import parseString | ||||
| from .utils import u_ | ||||
| 
 | ||||
| RDF_NAMESPACE = "http://www.w3.org/1999/02/22-rdf-syntax-ns#" | ||||
| DC_NAMESPACE = "http://purl.org/dc/elements/1.1/" | ||||
| XMP_NAMESPACE = "http://ns.adobe.com/xap/1.0/" | ||||
| PDF_NAMESPACE = "http://ns.adobe.com/pdf/1.3/" | ||||
| XMPMM_NAMESPACE = "http://ns.adobe.com/xap/1.0/mm/" | ||||
| 
 | ||||
| # What is the PDFX namespace, you might ask?  I might ask that too.  It's | ||||
| # a completely undocumented namespace used to place "custom metadata" | ||||
| # properties, which are arbitrary metadata properties with no semantic or | ||||
| # documented meaning.  Elements in the namespace are key/value-style storage, | ||||
| # where the element name is the key and the content is the value.  The keys | ||||
| # are transformed into valid XML identifiers by substituting an invalid | ||||
| # identifier character with \u2182 followed by the unicode hex ID of the | ||||
| # original character.  A key like "my car" is therefore "my\u21820020car". | ||||
| # | ||||
| # \u2182, in case you're wondering, is the unicode character | ||||
| # \u{ROMAN NUMERAL TEN THOUSAND}, a straightforward and obvious choice for | ||||
| # escaping characters. | ||||
| # | ||||
| # Intentional users of the pdfx namespace should be shot on sight.  A | ||||
| # custom data schema and sensical XML elements could be used instead, as is | ||||
| # suggested by Adobe's own documentation on XMP (under "Extensibility of | ||||
| # Schemas"). | ||||
| # | ||||
| # Information presented here on the /pdfx/ schema is a result of limited | ||||
| # reverse engineering, and does not constitute a full specification. | ||||
| PDFX_NAMESPACE = "http://ns.adobe.com/pdfx/1.3/" | ||||
| 
 | ||||
| iso8601 = re.compile(""" | ||||
|         (?P<year>[0-9]{4}) | ||||
|         (- | ||||
|             (?P<month>[0-9]{2}) | ||||
|             (- | ||||
|                 (?P<day>[0-9]+) | ||||
|                 (T | ||||
|                     (?P<hour>[0-9]{2}): | ||||
|                     (?P<minute>[0-9]{2}) | ||||
|                     (:(?P<second>[0-9]{2}(.[0-9]+)?))? | ||||
|                     (?P<tzd>Z|[-+][0-9]{2}:[0-9]{2}) | ||||
|                 )? | ||||
|             )? | ||||
|         )? | ||||
|         """, re.VERBOSE) | ||||
| 
 | ||||
| 
 | ||||
| class XmpInformation(PdfObject): | ||||
|     """ | ||||
|     An object that represents Adobe XMP metadata. | ||||
|     Usually accessed by :meth:`getXmpMetadata()<PyPDF2.PdfFileReader.getXmpMetadata>` | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, stream): | ||||
|         self.stream = stream | ||||
|         docRoot = parseString(self.stream.getData()) | ||||
|         self.rdfRoot = docRoot.getElementsByTagNameNS(RDF_NAMESPACE, "RDF")[0] | ||||
|         self.cache = {} | ||||
| 
 | ||||
|     def writeToStream(self, stream, encryption_key): | ||||
|         self.stream.writeToStream(stream, encryption_key) | ||||
| 
 | ||||
|     def getElement(self, aboutUri, namespace, name): | ||||
|         for desc in self.rdfRoot.getElementsByTagNameNS(RDF_NAMESPACE, "Description"): | ||||
|             if desc.getAttributeNS(RDF_NAMESPACE, "about") == aboutUri: | ||||
|                 attr = desc.getAttributeNodeNS(namespace, name) | ||||
|                 if attr != None: | ||||
|                     yield attr | ||||
|                 for element in desc.getElementsByTagNameNS(namespace, name): | ||||
|                     yield element | ||||
| 
 | ||||
|     def getNodesInNamespace(self, aboutUri, namespace): | ||||
|         for desc in self.rdfRoot.getElementsByTagNameNS(RDF_NAMESPACE, "Description"): | ||||
|             if desc.getAttributeNS(RDF_NAMESPACE, "about") == aboutUri: | ||||
|                 for i in range(desc.attributes.length): | ||||
|                     attr = desc.attributes.item(i) | ||||
|                     if attr.namespaceURI == namespace: | ||||
|                         yield attr | ||||
|                 for child in desc.childNodes: | ||||
|                     if child.namespaceURI == namespace: | ||||
|                         yield child | ||||
| 
 | ||||
|     def _getText(self, element): | ||||
|         text = "" | ||||
|         for child in element.childNodes: | ||||
|             if child.nodeType == child.TEXT_NODE: | ||||
|                 text += child.data | ||||
|         return text | ||||
| 
 | ||||
|     def _converter_string(value): | ||||
|         return value | ||||
| 
 | ||||
|     def _converter_date(value): | ||||
|         m = iso8601.match(value) | ||||
|         year = int(m.group("year")) | ||||
|         month = int(m.group("month") or "1") | ||||
|         day = int(m.group("day") or "1") | ||||
|         hour = int(m.group("hour") or "0") | ||||
|         minute = int(m.group("minute") or "0") | ||||
|         second = decimal.Decimal(m.group("second") or "0") | ||||
|         seconds = second.to_integral(decimal.ROUND_FLOOR) | ||||
|         milliseconds = (second - seconds) * 1000000 | ||||
|         tzd = m.group("tzd") or "Z" | ||||
|         dt = datetime.datetime(year, month, day, hour, minute, seconds, milliseconds) | ||||
|         if tzd != "Z": | ||||
|             tzd_hours, tzd_minutes = [int(x) for x in tzd.split(":")] | ||||
|             tzd_hours *= -1 | ||||
|             if tzd_hours < 0: | ||||
|                 tzd_minutes *= -1 | ||||
|             dt = dt + datetime.timedelta(hours=tzd_hours, minutes=tzd_minutes) | ||||
|         return dt | ||||
|     _test_converter_date = staticmethod(_converter_date) | ||||
| 
 | ||||
|     def _getter_bag(namespace, name, converter): | ||||
|         def get(self): | ||||
|             cached = self.cache.get(namespace, {}).get(name) | ||||
|             if cached: | ||||
|                 return cached | ||||
|             retval = [] | ||||
|             for element in self.getElement("", namespace, name): | ||||
|                 bags = element.getElementsByTagNameNS(RDF_NAMESPACE, "Bag") | ||||
|                 if len(bags): | ||||
|                     for bag in bags: | ||||
|                         for item in bag.getElementsByTagNameNS(RDF_NAMESPACE, "li"): | ||||
|                             value = self._getText(item) | ||||
|                             value = converter(value) | ||||
|                             retval.append(value) | ||||
|             ns_cache = self.cache.setdefault(namespace, {}) | ||||
|             ns_cache[name] = retval | ||||
|             return retval | ||||
|         return get | ||||
| 
 | ||||
|     def _getter_seq(namespace, name, converter): | ||||
|         def get(self): | ||||
|             cached = self.cache.get(namespace, {}).get(name) | ||||
|             if cached: | ||||
|                 return cached | ||||
|             retval = [] | ||||
|             for element in self.getElement("", namespace, name): | ||||
|                 seqs = element.getElementsByTagNameNS(RDF_NAMESPACE, "Seq") | ||||
|                 if len(seqs): | ||||
|                     for seq in seqs: | ||||
|                         for item in seq.getElementsByTagNameNS(RDF_NAMESPACE, "li"): | ||||
|                             value = self._getText(item) | ||||
|                             value = converter(value) | ||||
|                             retval.append(value) | ||||
|                 else: | ||||
|                     value = converter(self._getText(element)) | ||||
|                     retval.append(value) | ||||
|             ns_cache = self.cache.setdefault(namespace, {}) | ||||
|             ns_cache[name] = retval | ||||
|             return retval | ||||
|         return get | ||||
| 
 | ||||
|     def _getter_langalt(namespace, name, converter): | ||||
|         def get(self): | ||||
|             cached = self.cache.get(namespace, {}).get(name) | ||||
|             if cached: | ||||
|                 return cached | ||||
|             retval = {} | ||||
|             for element in self.getElement("", namespace, name): | ||||
|                 alts = element.getElementsByTagNameNS(RDF_NAMESPACE, "Alt") | ||||
|                 if len(alts): | ||||
|                     for alt in alts: | ||||
|                         for item in alt.getElementsByTagNameNS(RDF_NAMESPACE, "li"): | ||||
|                             value = self._getText(item) | ||||
|                             value = converter(value) | ||||
|                             retval[item.getAttribute("xml:lang")] = value | ||||
|                 else: | ||||
|                     retval["x-default"] = converter(self._getText(element)) | ||||
|             ns_cache = self.cache.setdefault(namespace, {}) | ||||
|             ns_cache[name] = retval | ||||
|             return retval | ||||
|         return get | ||||
| 
 | ||||
|     def _getter_single(namespace, name, converter): | ||||
|         def get(self): | ||||
|             cached = self.cache.get(namespace, {}).get(name) | ||||
|             if cached: | ||||
|                 return cached | ||||
|             value = None | ||||
|             for element in self.getElement("", namespace, name): | ||||
|                 if element.nodeType == element.ATTRIBUTE_NODE: | ||||
|                     value = element.nodeValue | ||||
|                 else: | ||||
|                     value = self._getText(element) | ||||
|                 break | ||||
|             if value != None: | ||||
|                 value = converter(value) | ||||
|             ns_cache = self.cache.setdefault(namespace, {}) | ||||
|             ns_cache[name] = value | ||||
|             return value | ||||
|         return get | ||||
| 
 | ||||
|     dc_contributor = property(_getter_bag(DC_NAMESPACE, "contributor", _converter_string)) | ||||
|     """ | ||||
|     Contributors to the resource (other than the authors). An unsorted | ||||
|     array of names. | ||||
|     """ | ||||
| 
 | ||||
|     dc_coverage = property(_getter_single(DC_NAMESPACE, "coverage", _converter_string)) | ||||
|     """ | ||||
|     Text describing the extent or scope of the resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_creator = property(_getter_seq(DC_NAMESPACE, "creator", _converter_string)) | ||||
|     """ | ||||
|     A sorted array of names of the authors of the resource, listed in order | ||||
|     of precedence. | ||||
|     """ | ||||
| 
 | ||||
|     dc_date = property(_getter_seq(DC_NAMESPACE, "date", _converter_date)) | ||||
|     """ | ||||
|     A sorted array of dates (datetime.datetime instances) of signifigance to | ||||
|     the resource.  The dates and times are in UTC. | ||||
|     """ | ||||
| 
 | ||||
|     dc_description = property(_getter_langalt(DC_NAMESPACE, "description", _converter_string)) | ||||
|     """ | ||||
|     A language-keyed dictionary of textual descriptions of the content of the | ||||
|     resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_format = property(_getter_single(DC_NAMESPACE, "format", _converter_string)) | ||||
|     """ | ||||
|     The mime-type of the resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_identifier = property(_getter_single(DC_NAMESPACE, "identifier", _converter_string)) | ||||
|     """ | ||||
|     Unique identifier of the resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_language = property(_getter_bag(DC_NAMESPACE, "language", _converter_string)) | ||||
|     """ | ||||
|     An unordered array specifying the languages used in the resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_publisher = property(_getter_bag(DC_NAMESPACE, "publisher", _converter_string)) | ||||
|     """ | ||||
|     An unordered array of publisher names. | ||||
|     """ | ||||
| 
 | ||||
|     dc_relation = property(_getter_bag(DC_NAMESPACE, "relation", _converter_string)) | ||||
|     """ | ||||
|     An unordered array of text descriptions of relationships to other | ||||
|     documents. | ||||
|     """ | ||||
| 
 | ||||
|     dc_rights = property(_getter_langalt(DC_NAMESPACE, "rights", _converter_string)) | ||||
|     """ | ||||
|     A language-keyed dictionary of textual descriptions of the rights the | ||||
|     user has to this resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_source = property(_getter_single(DC_NAMESPACE, "source", _converter_string)) | ||||
|     """ | ||||
|     Unique identifier of the work from which this resource was derived. | ||||
|     """ | ||||
| 
 | ||||
|     dc_subject = property(_getter_bag(DC_NAMESPACE, "subject", _converter_string)) | ||||
|     """ | ||||
|     An unordered array of descriptive phrases or keywrods that specify the | ||||
|     topic of the content of the resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_title = property(_getter_langalt(DC_NAMESPACE, "title", _converter_string)) | ||||
|     """ | ||||
|     A language-keyed dictionary of the title of the resource. | ||||
|     """ | ||||
| 
 | ||||
|     dc_type = property(_getter_bag(DC_NAMESPACE, "type", _converter_string)) | ||||
|     """ | ||||
|     An unordered array of textual descriptions of the document type. | ||||
|     """ | ||||
| 
 | ||||
|     pdf_keywords = property(_getter_single(PDF_NAMESPACE, "Keywords", _converter_string)) | ||||
|     """ | ||||
|     An unformatted text string representing document keywords. | ||||
|     """ | ||||
| 
 | ||||
|     pdf_pdfversion = property(_getter_single(PDF_NAMESPACE, "PDFVersion", _converter_string)) | ||||
|     """ | ||||
|     The PDF file version, for example 1.0, 1.3. | ||||
|     """ | ||||
| 
 | ||||
|     pdf_producer = property(_getter_single(PDF_NAMESPACE, "Producer", _converter_string)) | ||||
|     """ | ||||
|     The name of the tool that created the PDF document. | ||||
|     """ | ||||
| 
 | ||||
|     xmp_createDate = property(_getter_single(XMP_NAMESPACE, "CreateDate", _converter_date)) | ||||
|     """ | ||||
|     The date and time the resource was originally created.  The date and | ||||
|     time are returned as a UTC datetime.datetime object. | ||||
|     """ | ||||
| 
 | ||||
|     xmp_modifyDate = property(_getter_single(XMP_NAMESPACE, "ModifyDate", _converter_date)) | ||||
|     """ | ||||
|     The date and time the resource was last modified.  The date and time | ||||
|     are returned as a UTC datetime.datetime object. | ||||
|     """ | ||||
| 
 | ||||
|     xmp_metadataDate = property(_getter_single(XMP_NAMESPACE, "MetadataDate", _converter_date)) | ||||
|     """ | ||||
|     The date and time that any metadata for this resource was last | ||||
|     changed.  The date and time are returned as a UTC datetime.datetime | ||||
|     object. | ||||
|     """ | ||||
| 
 | ||||
|     xmp_creatorTool = property(_getter_single(XMP_NAMESPACE, "CreatorTool", _converter_string)) | ||||
|     """ | ||||
|     The name of the first known tool used to create the resource. | ||||
|     """ | ||||
| 
 | ||||
|     xmpmm_documentId = property(_getter_single(XMPMM_NAMESPACE, "DocumentID", _converter_string)) | ||||
|     """ | ||||
|     The common identifier for all versions and renditions of this resource. | ||||
|     """ | ||||
| 
 | ||||
|     xmpmm_instanceId = property(_getter_single(XMPMM_NAMESPACE, "InstanceID", _converter_string)) | ||||
|     """ | ||||
|     An identifier for a specific incarnation of a document, updated each | ||||
|     time a file is saved. | ||||
|     """ | ||||
| 
 | ||||
|     def custom_properties(self): | ||||
|         if not hasattr(self, "_custom_properties"): | ||||
|             self._custom_properties = {} | ||||
|             for node in self.getNodesInNamespace("", PDFX_NAMESPACE): | ||||
|                 key = node.localName | ||||
|                 while True: | ||||
|                     # see documentation about PDFX_NAMESPACE earlier in file | ||||
|                     idx = key.find(u_("\u2182")) | ||||
|                     if idx == -1: | ||||
|                         break | ||||
|                     key = key[:idx] + chr(int(key[idx+1:idx+5], base=16)) + key[idx+5:] | ||||
|                 if node.nodeType == node.ATTRIBUTE_NODE: | ||||
|                     value = node.nodeValue | ||||
|                 else: | ||||
|                     value = self._getText(node) | ||||
|                 self._custom_properties[key] = value | ||||
|         return self._custom_properties | ||||
| 
 | ||||
|     custom_properties = property(custom_properties) | ||||
|     """ | ||||
|     Retrieves custom metadata properties defined in the undocumented pdfx | ||||
|     metadata schema. | ||||
| 
 | ||||
|     :return: a dictionary of key/value items for custom metadata properties. | ||||
|     :rtype: dict | ||||
|     """ | ||||
							
								
								
									
										1
									
								
								vendor/_version.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								vendor/_version.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1 +0,0 @@ | |||
| __version__ = '5.0.6' | ||||
							
								
								
									
										28
									
								
								vendor/babel/AUTHORS
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								vendor/babel/AUTHORS
									
									
									
									
										vendored
									
									
								
							|  | @ -1,28 +0,0 @@ | |||
| Babel is written and maintained by the Babel team and various contributors: | ||||
| 
 | ||||
| Maintainer and Current Project Lead: | ||||
| 
 | ||||
| - Armin Ronacher <armin.ronacher@active-4.com> | ||||
| 
 | ||||
| Contributors: | ||||
| 
 | ||||
| - Christopher Lenz <cmlenz@gmail.com> | ||||
| - Alex Morega <alex@grep.ro> | ||||
| - Felix Schwarz <felix.schwarz@oss.schwarz.eu> | ||||
| - Pedro Algarvio <pedro@algarvio.me> | ||||
| - Jeroen Ruigrok van der Werven <asmodai@in-nomine.org> | ||||
| - Philip Jenvey <pjenvey@underboss.org> | ||||
| - Tobias Bieniek <Tobias.Bieniek@gmx.de> | ||||
| - Jonas Borgström <jonas@edgewall.org> | ||||
| - Daniel Neuhäuser <dasdasich@gmail.com> | ||||
| - Nick Retallack <nick@bitcasa.com> | ||||
| - Thomas Waldmann <tw@waldmann-edv.de> | ||||
| - Lennart Regebro <regebro@gmail.com> | ||||
| 
 | ||||
| Babel was previously developed under the Copyright of Edgewall Software.  The | ||||
| following copyright notice holds true for releases before 2013: "Copyright (c) | ||||
| 2007 - 2011 by Edgewall Software" | ||||
| 
 | ||||
| In addition to the regular contributions Babel includes a fork of Lennart | ||||
| Regebro's tzlocal that originally was licensed under the CC0 license.  The | ||||
| original copyright of that project is "Copyright 2013 by Lennart Regebro". | ||||
							
								
								
									
										29
									
								
								vendor/babel/LICENSE
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								vendor/babel/LICENSE
									
									
									
									
										vendored
									
									
								
							|  | @ -1,29 +0,0 @@ | |||
| Copyright (C) 2013 by the Babel Team, see AUTHORS for more information. | ||||
| 
 | ||||
| All rights reserved. | ||||
| 
 | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions | ||||
| are met: | ||||
| 
 | ||||
|  1. Redistributions of source code must retain the above copyright | ||||
|     notice, this list of conditions and the following disclaimer. | ||||
|  2. Redistributions in binary form must reproduce the above copyright | ||||
|     notice, this list of conditions and the following disclaimer in | ||||
|     the documentation and/or other materials provided with the | ||||
|     distribution. | ||||
|  3. The name of the author may not be used to endorse or promote | ||||
|     products derived from this software without specific prior | ||||
|     written permission. | ||||
| 
 | ||||
| THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS | ||||
| OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||||
| WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||
| ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY | ||||
| DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | ||||
| DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE | ||||
| GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||
| INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER | ||||
| IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR | ||||
| OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN | ||||
| IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
							
								
								
									
										24
									
								
								vendor/babel/__init__.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										24
									
								
								vendor/babel/__init__.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,24 +0,0 @@ | |||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
|     babel | ||||
|     ~~~~~ | ||||
| 
 | ||||
|     Integrated collection of utilities that assist in internationalizing and | ||||
|     localizing applications. | ||||
| 
 | ||||
|     This package is basically composed of two major parts: | ||||
| 
 | ||||
|      * tools to build and work with ``gettext`` message catalogs | ||||
|      * a Python interface to the CLDR (Common Locale Data Repository), providing | ||||
|        access to various locale display names, localized number and date | ||||
|        formatting, etc. | ||||
| 
 | ||||
|     :copyright: (c) 2013 by the Babel Team. | ||||
|     :license: BSD, see LICENSE for more details. | ||||
| """ | ||||
| 
 | ||||
| from babel.core import UnknownLocaleError, Locale, default_locale, \ | ||||
|      negotiate_locale, parse_locale, get_locale_identifier | ||||
| 
 | ||||
| 
 | ||||
| __version__ = '1.3' | ||||
							
								
								
									
										51
									
								
								vendor/babel/_compat.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										51
									
								
								vendor/babel/_compat.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,51 +0,0 @@ | |||
| import sys | ||||
| 
 | ||||
| PY2 = sys.version_info[0] == 2 | ||||
| 
 | ||||
| _identity = lambda x: x | ||||
| 
 | ||||
| 
 | ||||
| if not PY2: | ||||
|     text_type = str | ||||
|     string_types = (str,) | ||||
|     integer_types = (int, ) | ||||
|     unichr = chr | ||||
| 
 | ||||
|     text_to_native = lambda s, enc: s | ||||
| 
 | ||||
|     iterkeys = lambda d: iter(d.keys()) | ||||
|     itervalues = lambda d: iter(d.values()) | ||||
|     iteritems = lambda d: iter(d.items()) | ||||
| 
 | ||||
|     from io import StringIO, BytesIO | ||||
|     import pickle | ||||
| 
 | ||||
|     izip = zip | ||||
|     imap = map | ||||
|     range_type = range | ||||
| 
 | ||||
|     cmp = lambda a, b: (a > b) - (a < b) | ||||
| 
 | ||||
| else: | ||||
|     text_type = unicode | ||||
|     string_types = (str, unicode) | ||||
|     integer_types = (int, long) | ||||
| 
 | ||||
|     text_to_native = lambda s, enc: s.encode(enc) | ||||
|     unichr = unichr | ||||
| 
 | ||||
|     iterkeys = lambda d: d.iterkeys() | ||||
|     itervalues = lambda d: d.itervalues() | ||||
|     iteritems = lambda d: d.iteritems() | ||||
| 
 | ||||
|     from cStringIO import StringIO as BytesIO | ||||
|     from StringIO import StringIO | ||||
|     import cPickle as pickle | ||||
| 
 | ||||
|     from itertools import izip, imap | ||||
|     range_type = xrange | ||||
| 
 | ||||
|     cmp = cmp | ||||
| 
 | ||||
| 
 | ||||
| number_types = integer_types + (float,) | ||||
							
								
								
									
										941
									
								
								vendor/babel/core.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										941
									
								
								vendor/babel/core.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,941 +0,0 @@ | |||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
|     babel.core | ||||
|     ~~~~~~~~~~ | ||||
| 
 | ||||
|     Core locale representation and locale data access. | ||||
| 
 | ||||
|     :copyright: (c) 2013 by the Babel Team. | ||||
|     :license: BSD, see LICENSE for more details. | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| 
 | ||||
| from babel import localedata | ||||
| from babel._compat import pickle, string_types | ||||
| 
 | ||||
| __all__ = ['UnknownLocaleError', 'Locale', 'default_locale', 'negotiate_locale', | ||||
|            'parse_locale'] | ||||
| 
 | ||||
| 
 | ||||
| _global_data = None | ||||
| 
 | ||||
| 
 | ||||
| def _raise_no_data_error(): | ||||
|     raise RuntimeError('The babel data files are not available. ' | ||||
|                        'This usually happens because you are using ' | ||||
|                        'a source checkout from Babel and you did ' | ||||
|                        'not build the data files.  Just make sure ' | ||||
|                        'to run "python setup.py import_cldr" before ' | ||||
|                        'installing the library.') | ||||
| 
 | ||||
| 
 | ||||
| def get_global(key): | ||||
|     """Return the dictionary for the given key in the global data. | ||||
| 
 | ||||
|     The global data is stored in the ``babel/global.dat`` file and contains | ||||
|     information independent of individual locales. | ||||
| 
 | ||||
|     >>> get_global('zone_aliases')['UTC'] | ||||
|     u'Etc/GMT' | ||||
|     >>> get_global('zone_territories')['Europe/Berlin'] | ||||
|     u'DE' | ||||
| 
 | ||||
|     .. versionadded:: 0.9 | ||||
| 
 | ||||
|     :param key: the data key | ||||
|     """ | ||||
|     global _global_data | ||||
|     if _global_data is None: | ||||
|         dirname = os.path.join(os.path.dirname(__file__)) | ||||
|         filename = os.path.join(dirname, 'global.dat') | ||||
|         if not os.path.isfile(filename): | ||||
|             _raise_no_data_error() | ||||
|         fileobj = open(filename, 'rb') | ||||
|         try: | ||||
|             _global_data = pickle.load(fileobj) | ||||
|         finally: | ||||
|             fileobj.close() | ||||
|     return _global_data.get(key, {}) | ||||
| 
 | ||||
| 
 | ||||
| LOCALE_ALIASES = { | ||||
|     'ar': 'ar_SY', 'bg': 'bg_BG', 'bs': 'bs_BA', 'ca': 'ca_ES', 'cs': 'cs_CZ', | ||||
|     'da': 'da_DK', 'de': 'de_DE', 'el': 'el_GR', 'en': 'en_US', 'es': 'es_ES', | ||||
|     'et': 'et_EE', 'fa': 'fa_IR', 'fi': 'fi_FI', 'fr': 'fr_FR', 'gl': 'gl_ES', | ||||
|     'he': 'he_IL', 'hu': 'hu_HU', 'id': 'id_ID', 'is': 'is_IS', 'it': 'it_IT', | ||||
|     'ja': 'ja_JP', 'km': 'km_KH', 'ko': 'ko_KR', 'lt': 'lt_LT', 'lv': 'lv_LV', | ||||
|     'mk': 'mk_MK', 'nl': 'nl_NL', 'nn': 'nn_NO', 'no': 'nb_NO', 'pl': 'pl_PL', | ||||
|     'pt': 'pt_PT', 'ro': 'ro_RO', 'ru': 'ru_RU', 'sk': 'sk_SK', 'sl': 'sl_SI', | ||||
|     'sv': 'sv_SE', 'th': 'th_TH', 'tr': 'tr_TR', 'uk': 'uk_UA' | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| class UnknownLocaleError(Exception): | ||||
|     """Exception thrown when a locale is requested for which no locale data | ||||
|     is available. | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, identifier): | ||||
|         """Create the exception. | ||||
| 
 | ||||
|         :param identifier: the identifier string of the unsupported locale | ||||
|         """ | ||||
|         Exception.__init__(self, 'unknown locale %r' % identifier) | ||||
| 
 | ||||
|         #: The identifier of the locale that could not be found. | ||||
|         self.identifier = identifier | ||||
| 
 | ||||
| 
 | ||||
| class Locale(object): | ||||
|     """Representation of a specific locale. | ||||
| 
 | ||||
|     >>> locale = Locale('en', 'US') | ||||
|     >>> repr(locale) | ||||
|     "Locale('en', territory='US')" | ||||
|     >>> locale.display_name | ||||
|     u'English (United States)' | ||||
| 
 | ||||
|     A `Locale` object can also be instantiated from a raw locale string: | ||||
| 
 | ||||
|     >>> locale = Locale.parse('en-US', sep='-') | ||||
|     >>> repr(locale) | ||||
|     "Locale('en', territory='US')" | ||||
| 
 | ||||
|     `Locale` objects provide access to a collection of locale data, such as | ||||
|     territory and language names, number and date format patterns, and more: | ||||
| 
 | ||||
|     >>> locale.number_symbols['decimal'] | ||||
|     u'.' | ||||
| 
 | ||||
|     If a locale is requested for which no locale data is available, an | ||||
|     `UnknownLocaleError` is raised: | ||||
| 
 | ||||
|     >>> Locale.parse('en_DE') | ||||
|     Traceback (most recent call last): | ||||
|         ... | ||||
|     UnknownLocaleError: unknown locale 'en_DE' | ||||
| 
 | ||||
|     For more information see :rfc:`3066`. | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, language, territory=None, script=None, variant=None): | ||||
|         """Initialize the locale object from the given identifier components. | ||||
| 
 | ||||
|         >>> locale = Locale('en', 'US') | ||||
|         >>> locale.language | ||||
|         'en' | ||||
|         >>> locale.territory | ||||
|         'US' | ||||
| 
 | ||||
|         :param language: the language code | ||||
|         :param territory: the territory (country or region) code | ||||
|         :param script: the script code | ||||
|         :param variant: the variant code | ||||
|         :raise `UnknownLocaleError`: if no locale data is available for the | ||||
|                                      requested locale | ||||
|         """ | ||||
|         #: the language code | ||||
|         self.language = language | ||||
|         #: the territory (country or region) code | ||||
|         self.territory = territory | ||||
|         #: the script code | ||||
|         self.script = script | ||||
|         #: the variant code | ||||
|         self.variant = variant | ||||
|         self.__data = None | ||||
| 
 | ||||
|         identifier = str(self) | ||||
|         if not localedata.exists(identifier): | ||||
|             raise UnknownLocaleError(identifier) | ||||
| 
 | ||||
|     @classmethod | ||||
|     def default(cls, category=None, aliases=LOCALE_ALIASES): | ||||
|         """Return the system default locale for the specified category. | ||||
| 
 | ||||
|         >>> for name in ['LANGUAGE', 'LC_ALL', 'LC_CTYPE', 'LC_MESSAGES']: | ||||
|         ...     os.environ[name] = '' | ||||
|         >>> os.environ['LANG'] = 'fr_FR.UTF-8' | ||||
|         >>> Locale.default('LC_MESSAGES') | ||||
|         Locale('fr', territory='FR') | ||||
| 
 | ||||
|         The following fallbacks to the variable are always considered: | ||||
| 
 | ||||
|         - ``LANGUAGE`` | ||||
|         - ``LC_ALL`` | ||||
|         - ``LC_CTYPE`` | ||||
|         - ``LANG`` | ||||
| 
 | ||||
|         :param category: one of the ``LC_XXX`` environment variable names | ||||
|         :param aliases: a dictionary of aliases for locale identifiers | ||||
|         """ | ||||
|         # XXX: use likely subtag expansion here instead of the | ||||
|         # aliases dictionary. | ||||
|         locale_string = default_locale(category, aliases=aliases) | ||||
|         return cls.parse(locale_string) | ||||
| 
 | ||||
|     @classmethod | ||||
|     def negotiate(cls, preferred, available, sep='_', aliases=LOCALE_ALIASES): | ||||
|         """Find the best match between available and requested locale strings. | ||||
| 
 | ||||
|         >>> Locale.negotiate(['de_DE', 'en_US'], ['de_DE', 'de_AT']) | ||||
|         Locale('de', territory='DE') | ||||
|         >>> Locale.negotiate(['de_DE', 'en_US'], ['en', 'de']) | ||||
|         Locale('de') | ||||
|         >>> Locale.negotiate(['de_DE', 'de'], ['en_US']) | ||||
| 
 | ||||
|         You can specify the character used in the locale identifiers to separate | ||||
|         the differnet components. This separator is applied to both lists. Also, | ||||
|         case is ignored in the comparison: | ||||
| 
 | ||||
|         >>> Locale.negotiate(['de-DE', 'de'], ['en-us', 'de-de'], sep='-') | ||||
|         Locale('de', territory='DE') | ||||
| 
 | ||||
|         :param preferred: the list of locale identifers preferred by the user | ||||
|         :param available: the list of locale identifiers available | ||||
|         :param aliases: a dictionary of aliases for locale identifiers | ||||
|         """ | ||||
|         identifier = negotiate_locale(preferred, available, sep=sep, | ||||
|                                       aliases=aliases) | ||||
|         if identifier: | ||||
|             return Locale.parse(identifier, sep=sep) | ||||
| 
 | ||||
|     @classmethod | ||||
|     def parse(cls, identifier, sep='_', resolve_likely_subtags=True): | ||||
|         """Create a `Locale` instance for the given locale identifier. | ||||
| 
 | ||||
|         >>> l = Locale.parse('de-DE', sep='-') | ||||
|         >>> l.display_name | ||||
|         u'Deutsch (Deutschland)' | ||||
| 
 | ||||
|         If the `identifier` parameter is not a string, but actually a `Locale` | ||||
|         object, that object is returned: | ||||
| 
 | ||||
|         >>> Locale.parse(l) | ||||
|         Locale('de', territory='DE') | ||||
| 
 | ||||
|         This also can perform resolving of likely subtags which it does | ||||
|         by default.  This is for instance useful to figure out the most | ||||
|         likely locale for a territory you can use ``'und'`` as the | ||||
|         language tag: | ||||
| 
 | ||||
|         >>> Locale.parse('und_AT') | ||||
|         Locale('de', territory='AT') | ||||
| 
 | ||||
|         :param identifier: the locale identifier string | ||||
|         :param sep: optional component separator | ||||
|         :param resolve_likely_subtags: if this is specified then a locale will | ||||
|                                        have its likely subtag resolved if the | ||||
|                                        locale otherwise does not exist.  For | ||||
|                                        instance ``zh_TW`` by itself is not a | ||||
|                                        locale that exists but Babel can | ||||
|                                        automatically expand it to the full | ||||
|                                        form of ``zh_hant_TW``.  Note that this | ||||
|                                        expansion is only taking place if no | ||||
|                                        locale exists otherwise.  For instance | ||||
|                                        there is a locale ``en`` that can exist | ||||
|                                        by itself. | ||||
|         :raise `ValueError`: if the string does not appear to be a valid locale | ||||
|                              identifier | ||||
|         :raise `UnknownLocaleError`: if no locale data is available for the | ||||
|                                      requested locale | ||||
|         """ | ||||
|         if identifier is None: | ||||
|             return None | ||||
|         elif isinstance(identifier, Locale): | ||||
|             return identifier | ||||
|         elif not isinstance(identifier, string_types): | ||||
|             raise TypeError('Unxpected value for identifier: %r' % (identifier,)) | ||||
| 
 | ||||
|         parts = parse_locale(identifier, sep=sep) | ||||
|         input_id = get_locale_identifier(parts) | ||||
| 
 | ||||
|         def _try_load(parts): | ||||
|             try: | ||||
|                 return cls(*parts) | ||||
|             except UnknownLocaleError: | ||||
|                 return None | ||||
| 
 | ||||
|         def _try_load_reducing(parts): | ||||
|             # Success on first hit, return it. | ||||
|             locale = _try_load(parts) | ||||
|             if locale is not None: | ||||
|                 return locale | ||||
| 
 | ||||
|             # Now try without script and variant | ||||
|             locale = _try_load(parts[:2]) | ||||
|             if locale is not None: | ||||
|                 return locale | ||||
| 
 | ||||
|         locale = _try_load(parts) | ||||
|         if locale is not None: | ||||
|             return locale | ||||
|         if not resolve_likely_subtags: | ||||
|             raise UnknownLocaleError(input_id) | ||||
| 
 | ||||
|         # From here onwards is some very bad likely subtag resolving.  This | ||||
|         # whole logic is not entirely correct but good enough (tm) for the | ||||
|         # time being.  This has been added so that zh_TW does not cause | ||||
|         # errors for people when they upgrade.  Later we should properly | ||||
|         # implement ICU like fuzzy locale objects and provide a way to | ||||
|         # maximize and minimize locale tags. | ||||
| 
 | ||||
|         language, territory, script, variant = parts | ||||
|         language = get_global('language_aliases').get(language, language) | ||||
|         territory = get_global('territory_aliases').get(territory, territory) | ||||
|         script = get_global('script_aliases').get(script, script) | ||||
|         variant = get_global('variant_aliases').get(variant, variant) | ||||
| 
 | ||||
|         if territory == 'ZZ': | ||||
|             territory = None | ||||
|         if script == 'Zzzz': | ||||
|             script = None | ||||
| 
 | ||||
|         parts = language, territory, script, variant | ||||
| 
 | ||||
|         # First match: try the whole identifier | ||||
|         new_id = get_locale_identifier(parts) | ||||
|         likely_subtag = get_global('likely_subtags').get(new_id) | ||||
|         if likely_subtag is not None: | ||||
|             locale = _try_load_reducing(parse_locale(likely_subtag)) | ||||
|             if locale is not None: | ||||
|                 return locale | ||||
| 
 | ||||
|         # If we did not find anything so far, try again with a | ||||
|         # simplified identifier that is just the language | ||||
|         likely_subtag = get_global('likely_subtags').get(language) | ||||
|         if likely_subtag is not None: | ||||
|             language2, _, script2, variant2 = parse_locale(likely_subtag) | ||||
|             locale = _try_load_reducing((language2, territory, script2, variant2)) | ||||
|             if locale is not None: | ||||
|                 return locale | ||||
| 
 | ||||
|         raise UnknownLocaleError(input_id) | ||||
| 
 | ||||
|     def __eq__(self, other): | ||||
|         for key in ('language', 'territory', 'script', 'variant'): | ||||
|             if not hasattr(other, key): | ||||
|                 return False | ||||
|         return (self.language == other.language) and \ | ||||
|             (self.territory == other.territory) and \ | ||||
|             (self.script == other.script) and \ | ||||
|             (self.variant == other.variant) | ||||
| 
 | ||||
|     def __ne__(self, other): | ||||
|         return not self.__eq__(other) | ||||
| 
 | ||||
|     def __repr__(self): | ||||
|         parameters = [''] | ||||
|         for key in ('territory', 'script', 'variant'): | ||||
|             value = getattr(self, key) | ||||
|             if value is not None: | ||||
|                 parameters.append('%s=%r' % (key, value)) | ||||
|         parameter_string = '%r' % self.language + ', '.join(parameters) | ||||
|         return 'Locale(%s)' % parameter_string | ||||
| 
 | ||||
|     def __str__(self): | ||||
|         return get_locale_identifier((self.language, self.territory, | ||||
|                                       self.script, self.variant)) | ||||
| 
 | ||||
|     @property | ||||
|     def _data(self): | ||||
|         if self.__data is None: | ||||
|             self.__data = localedata.LocaleDataDict(localedata.load(str(self))) | ||||
|         return self.__data | ||||
| 
 | ||||
|     def get_display_name(self, locale=None): | ||||
|         """Return the display name of the locale using the given locale. | ||||
| 
 | ||||
|         The display name will include the language, territory, script, and | ||||
|         variant, if those are specified. | ||||
| 
 | ||||
|         >>> Locale('zh', 'CN', script='Hans').get_display_name('en') | ||||
|         u'Chinese (Simplified, China)' | ||||
| 
 | ||||
|         :param locale: the locale to use | ||||
|         """ | ||||
|         if locale is None: | ||||
|             locale = self | ||||
|         locale = Locale.parse(locale) | ||||
|         retval = locale.languages.get(self.language) | ||||
|         if self.territory or self.script or self.variant: | ||||
|             details = [] | ||||
|             if self.script: | ||||
|                 details.append(locale.scripts.get(self.script)) | ||||
|             if self.territory: | ||||
|                 details.append(locale.territories.get(self.territory)) | ||||
|             if self.variant: | ||||
|                 details.append(locale.variants.get(self.variant)) | ||||
|             details = filter(None, details) | ||||
|             if details: | ||||
|                 retval += ' (%s)' % u', '.join(details) | ||||
|         return retval | ||||
| 
 | ||||
|     display_name = property(get_display_name, doc="""\ | ||||
|         The localized display name of the locale. | ||||
| 
 | ||||
|         >>> Locale('en').display_name | ||||
|         u'English' | ||||
|         >>> Locale('en', 'US').display_name | ||||
|         u'English (United States)' | ||||
|         >>> Locale('sv').display_name | ||||
|         u'svenska' | ||||
| 
 | ||||
|         :type: `unicode` | ||||
|         """) | ||||
| 
 | ||||
|     def get_language_name(self, locale=None): | ||||
|         """Return the language of this locale in the given locale. | ||||
| 
 | ||||
|         >>> Locale('zh', 'CN', script='Hans').get_language_name('de') | ||||
|         u'Chinesisch' | ||||
| 
 | ||||
|         .. versionadded:: 1.0 | ||||
| 
 | ||||
|         :param locale: the locale to use | ||||
|         """ | ||||
|         if locale is None: | ||||
|             locale = self | ||||
|         locale = Locale.parse(locale) | ||||
|         return locale.languages.get(self.language) | ||||
| 
 | ||||
|     language_name = property(get_language_name, doc="""\ | ||||
|         The localized language name of the locale. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').language_name | ||||
|         u'English' | ||||
|     """) | ||||
| 
 | ||||
|     def get_territory_name(self, locale=None): | ||||
|         """Return the territory name in the given locale.""" | ||||
|         if locale is None: | ||||
|             locale = self | ||||
|         locale = Locale.parse(locale) | ||||
|         return locale.territories.get(self.territory) | ||||
| 
 | ||||
|     territory_name = property(get_territory_name, doc="""\ | ||||
|         The localized territory name of the locale if available. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').territory_name | ||||
|         u'Deutschland' | ||||
|     """) | ||||
| 
 | ||||
|     def get_script_name(self, locale=None): | ||||
|         """Return the script name in the given locale.""" | ||||
|         if locale is None: | ||||
|             locale = self | ||||
|         locale = Locale.parse(locale) | ||||
|         return locale.scripts.get(self.script) | ||||
| 
 | ||||
|     script_name = property(get_script_name, doc="""\ | ||||
|         The localized script name of the locale if available. | ||||
| 
 | ||||
|         >>> Locale('ms', 'SG', script='Latn').script_name | ||||
|         u'Latin' | ||||
|     """) | ||||
| 
 | ||||
|     @property | ||||
|     def english_name(self): | ||||
|         """The english display name of the locale. | ||||
| 
 | ||||
|         >>> Locale('de').english_name | ||||
|         u'German' | ||||
|         >>> Locale('de', 'DE').english_name | ||||
|         u'German (Germany)' | ||||
| 
 | ||||
|         :type: `unicode`""" | ||||
|         return self.get_display_name(Locale('en')) | ||||
| 
 | ||||
|     #{ General Locale Display Names | ||||
| 
 | ||||
|     @property | ||||
|     def languages(self): | ||||
|         """Mapping of language codes to translated language names. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').languages['ja'] | ||||
|         u'Japanisch' | ||||
| 
 | ||||
|         See `ISO 639 <http://www.loc.gov/standards/iso639-2/>`_ for | ||||
|         more information. | ||||
|         """ | ||||
|         return self._data['languages'] | ||||
| 
 | ||||
|     @property | ||||
|     def scripts(self): | ||||
|         """Mapping of script codes to translated script names. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').scripts['Hira'] | ||||
|         u'Hiragana' | ||||
| 
 | ||||
|         See `ISO 15924 <http://www.evertype.com/standards/iso15924/>`_ | ||||
|         for more information. | ||||
|         """ | ||||
|         return self._data['scripts'] | ||||
| 
 | ||||
|     @property | ||||
|     def territories(self): | ||||
|         """Mapping of script codes to translated script names. | ||||
| 
 | ||||
|         >>> Locale('es', 'CO').territories['DE'] | ||||
|         u'Alemania' | ||||
| 
 | ||||
|         See `ISO 3166 <http://www.iso.org/iso/en/prods-services/iso3166ma/>`_ | ||||
|         for more information. | ||||
|         """ | ||||
|         return self._data['territories'] | ||||
| 
 | ||||
|     @property | ||||
|     def variants(self): | ||||
|         """Mapping of script codes to translated script names. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').variants['1901'] | ||||
|         u'Alte deutsche Rechtschreibung' | ||||
|         """ | ||||
|         return self._data['variants'] | ||||
| 
 | ||||
|     #{ Number Formatting | ||||
| 
 | ||||
|     @property | ||||
|     def currencies(self): | ||||
|         """Mapping of currency codes to translated currency names.  This | ||||
|         only returns the generic form of the currency name, not the count | ||||
|         specific one.  If an actual number is requested use the | ||||
|         :func:`babel.numbers.get_currency_name` function. | ||||
| 
 | ||||
|         >>> Locale('en').currencies['COP'] | ||||
|         u'Colombian Peso' | ||||
|         >>> Locale('de', 'DE').currencies['COP'] | ||||
|         u'Kolumbianischer Peso' | ||||
|         """ | ||||
|         return self._data['currency_names'] | ||||
| 
 | ||||
|     @property | ||||
|     def currency_symbols(self): | ||||
|         """Mapping of currency codes to symbols. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').currency_symbols['USD'] | ||||
|         u'$' | ||||
|         >>> Locale('es', 'CO').currency_symbols['USD'] | ||||
|         u'US$' | ||||
|         """ | ||||
|         return self._data['currency_symbols'] | ||||
| 
 | ||||
|     @property | ||||
|     def number_symbols(self): | ||||
|         """Symbols used in number formatting. | ||||
| 
 | ||||
|         >>> Locale('fr', 'FR').number_symbols['decimal'] | ||||
|         u',' | ||||
|         """ | ||||
|         return self._data['number_symbols'] | ||||
| 
 | ||||
|     @property | ||||
|     def decimal_formats(self): | ||||
|         """Locale patterns for decimal number formatting. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').decimal_formats[None] | ||||
|         <NumberPattern u'#,##0.###'> | ||||
|         """ | ||||
|         return self._data['decimal_formats'] | ||||
| 
 | ||||
|     @property | ||||
|     def currency_formats(self): | ||||
|         """Locale patterns for currency number formatting. | ||||
| 
 | ||||
|         >>> print Locale('en', 'US').currency_formats[None] | ||||
|         <NumberPattern u'\\xa4#,##0.00'> | ||||
|         """ | ||||
|         return self._data['currency_formats'] | ||||
| 
 | ||||
|     @property | ||||
|     def percent_formats(self): | ||||
|         """Locale patterns for percent number formatting. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').percent_formats[None] | ||||
|         <NumberPattern u'#,##0%'> | ||||
|         """ | ||||
|         return self._data['percent_formats'] | ||||
| 
 | ||||
|     @property | ||||
|     def scientific_formats(self): | ||||
|         """Locale patterns for scientific number formatting. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').scientific_formats[None] | ||||
|         <NumberPattern u'#E0'> | ||||
|         """ | ||||
|         return self._data['scientific_formats'] | ||||
| 
 | ||||
|     #{ Calendar Information and Date Formatting | ||||
| 
 | ||||
|     @property | ||||
|     def periods(self): | ||||
|         """Locale display names for day periods (AM/PM). | ||||
| 
 | ||||
|         >>> Locale('en', 'US').periods['am'] | ||||
|         u'AM' | ||||
|         """ | ||||
|         return self._data['periods'] | ||||
| 
 | ||||
|     @property | ||||
|     def days(self): | ||||
|         """Locale display names for weekdays. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').days['format']['wide'][3] | ||||
|         u'Donnerstag' | ||||
|         """ | ||||
|         return self._data['days'] | ||||
| 
 | ||||
|     @property | ||||
|     def months(self): | ||||
|         """Locale display names for months. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').months['format']['wide'][10] | ||||
|         u'Oktober' | ||||
|         """ | ||||
|         return self._data['months'] | ||||
| 
 | ||||
|     @property | ||||
|     def quarters(self): | ||||
|         """Locale display names for quarters. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').quarters['format']['wide'][1] | ||||
|         u'1. Quartal' | ||||
|         """ | ||||
|         return self._data['quarters'] | ||||
| 
 | ||||
|     @property | ||||
|     def eras(self): | ||||
|         """Locale display names for eras. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').eras['wide'][1] | ||||
|         u'Anno Domini' | ||||
|         >>> Locale('en', 'US').eras['abbreviated'][0] | ||||
|         u'BC' | ||||
|         """ | ||||
|         return self._data['eras'] | ||||
| 
 | ||||
|     @property | ||||
|     def time_zones(self): | ||||
|         """Locale display names for time zones. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').time_zones['Europe/London']['long']['daylight'] | ||||
|         u'British Summer Time' | ||||
|         >>> Locale('en', 'US').time_zones['America/St_Johns']['city'] | ||||
|         u'St. John\u2019s' | ||||
|         """ | ||||
|         return self._data['time_zones'] | ||||
| 
 | ||||
|     @property | ||||
|     def meta_zones(self): | ||||
|         """Locale display names for meta time zones. | ||||
| 
 | ||||
|         Meta time zones are basically groups of different Olson time zones that | ||||
|         have the same GMT offset and daylight savings time. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').meta_zones['Europe_Central']['long']['daylight'] | ||||
|         u'Central European Summer Time' | ||||
| 
 | ||||
|         .. versionadded:: 0.9 | ||||
|         """ | ||||
|         return self._data['meta_zones'] | ||||
| 
 | ||||
|     @property | ||||
|     def zone_formats(self): | ||||
|         """Patterns related to the formatting of time zones. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').zone_formats['fallback'] | ||||
|         u'%(1)s (%(0)s)' | ||||
|         >>> Locale('pt', 'BR').zone_formats['region'] | ||||
|         u'Hor\\xe1rio %s' | ||||
| 
 | ||||
|         .. versionadded:: 0.9 | ||||
|         """ | ||||
|         return self._data['zone_formats'] | ||||
| 
 | ||||
|     @property | ||||
|     def first_week_day(self): | ||||
|         """The first day of a week, with 0 being Monday. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').first_week_day | ||||
|         0 | ||||
|         >>> Locale('en', 'US').first_week_day | ||||
|         6 | ||||
|         """ | ||||
|         return self._data['week_data']['first_day'] | ||||
| 
 | ||||
|     @property | ||||
|     def weekend_start(self): | ||||
|         """The day the weekend starts, with 0 being Monday. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').weekend_start | ||||
|         5 | ||||
|         """ | ||||
|         return self._data['week_data']['weekend_start'] | ||||
| 
 | ||||
|     @property | ||||
|     def weekend_end(self): | ||||
|         """The day the weekend ends, with 0 being Monday. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').weekend_end | ||||
|         6 | ||||
|         """ | ||||
|         return self._data['week_data']['weekend_end'] | ||||
| 
 | ||||
|     @property | ||||
|     def min_week_days(self): | ||||
|         """The minimum number of days in a week so that the week is counted as | ||||
|         the first week of a year or month. | ||||
| 
 | ||||
|         >>> Locale('de', 'DE').min_week_days | ||||
|         4 | ||||
|         """ | ||||
|         return self._data['week_data']['min_days'] | ||||
| 
 | ||||
|     @property | ||||
|     def date_formats(self): | ||||
|         """Locale patterns for date formatting. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').date_formats['short'] | ||||
|         <DateTimePattern u'M/d/yy'> | ||||
|         >>> Locale('fr', 'FR').date_formats['long'] | ||||
|         <DateTimePattern u'd MMMM y'> | ||||
|         """ | ||||
|         return self._data['date_formats'] | ||||
| 
 | ||||
|     @property | ||||
|     def time_formats(self): | ||||
|         """Locale patterns for time formatting. | ||||
| 
 | ||||
|         >>> Locale('en', 'US').time_formats['short'] | ||||
|         <DateTimePattern u'h:mm a'> | ||||
|         >>> Locale('fr', 'FR').time_formats['long'] | ||||
|         <DateTimePattern u'HH:mm:ss z'> | ||||
|         """ | ||||
|         return self._data['time_formats'] | ||||
| 
 | ||||
|     @property | ||||
|     def datetime_formats(self): | ||||
|         """Locale patterns for datetime formatting. | ||||
| 
 | ||||
|         >>> Locale('en').datetime_formats['full'] | ||||
|         u"{1} 'at' {0}" | ||||
|         >>> Locale('th').datetime_formats['medium'] | ||||
|         u'{1}, {0}' | ||||
|         """ | ||||
|         return self._data['datetime_formats'] | ||||
| 
 | ||||
|     @property | ||||
|     def plural_form(self): | ||||
|         """Plural rules for the locale. | ||||
| 
 | ||||
|         >>> Locale('en').plural_form(1) | ||||
|         'one' | ||||
|         >>> Locale('en').plural_form(0) | ||||
|         'other' | ||||
|         >>> Locale('fr').plural_form(0) | ||||
|         'one' | ||||
|         >>> Locale('ru').plural_form(100) | ||||
|         'many' | ||||
|         """ | ||||
|         return self._data['plural_form'] | ||||
| 
 | ||||
| 
 | ||||
| def default_locale(category=None, aliases=LOCALE_ALIASES): | ||||
|     """Returns the system default locale for a given category, based on | ||||
|     environment variables. | ||||
| 
 | ||||
|     >>> for name in ['LANGUAGE', 'LC_ALL', 'LC_CTYPE']: | ||||
|     ...     os.environ[name] = '' | ||||
|     >>> os.environ['LANG'] = 'fr_FR.UTF-8' | ||||
|     >>> default_locale('LC_MESSAGES') | ||||
|     'fr_FR' | ||||
| 
 | ||||
|     The "C" or "POSIX" pseudo-locales are treated as aliases for the | ||||
|     "en_US_POSIX" locale: | ||||
| 
 | ||||
|     >>> os.environ['LC_MESSAGES'] = 'POSIX' | ||||
|     >>> default_locale('LC_MESSAGES') | ||||
|     'en_US_POSIX' | ||||
| 
 | ||||
|     The following fallbacks to the variable are always considered: | ||||
| 
 | ||||
|     - ``LANGUAGE`` | ||||
|     - ``LC_ALL`` | ||||
|     - ``LC_CTYPE`` | ||||
|     - ``LANG`` | ||||
| 
 | ||||
|     :param category: one of the ``LC_XXX`` environment variable names | ||||
|     :param aliases: a dictionary of aliases for locale identifiers | ||||
|     """ | ||||
|     varnames = (category, 'LANGUAGE', 'LC_ALL', 'LC_CTYPE', 'LANG') | ||||
|     for name in filter(None, varnames): | ||||
|         locale = os.getenv(name) | ||||
|         if locale: | ||||
|             if name == 'LANGUAGE' and ':' in locale: | ||||
|                 # the LANGUAGE variable may contain a colon-separated list of | ||||
|                 # language codes; we just pick the language on the list | ||||
|                 locale = locale.split(':')[0] | ||||
|             if locale in ('C', 'POSIX'): | ||||
|                 locale = 'en_US_POSIX' | ||||
|             elif aliases and locale in aliases: | ||||
|                 locale = aliases[locale] | ||||
|             try: | ||||
|                 return get_locale_identifier(parse_locale(locale)) | ||||
|             except ValueError: | ||||
|                 pass | ||||
| 
 | ||||
| 
 | ||||
| def negotiate_locale(preferred, available, sep='_', aliases=LOCALE_ALIASES): | ||||
|     """Find the best match between available and requested locale strings. | ||||
| 
 | ||||
|     >>> negotiate_locale(['de_DE', 'en_US'], ['de_DE', 'de_AT']) | ||||
|     'de_DE' | ||||
|     >>> negotiate_locale(['de_DE', 'en_US'], ['en', 'de']) | ||||
|     'de' | ||||
| 
 | ||||
|     Case is ignored by the algorithm, the result uses the case of the preferred | ||||
|     locale identifier: | ||||
| 
 | ||||
|     >>> negotiate_locale(['de_DE', 'en_US'], ['de_de', 'de_at']) | ||||
|     'de_DE' | ||||
| 
 | ||||
|     >>> negotiate_locale(['de_DE', 'en_US'], ['de_de', 'de_at']) | ||||
|     'de_DE' | ||||
| 
 | ||||
|     By default, some web browsers unfortunately do not include the territory | ||||
|     in the locale identifier for many locales, and some don't even allow the | ||||
|     user to easily add the territory. So while you may prefer using qualified | ||||
|     locale identifiers in your web-application, they would not normally match | ||||
|     the language-only locale sent by such browsers. To workaround that, this | ||||
|     function uses a default mapping of commonly used langauge-only locale | ||||
|     identifiers to identifiers including the territory: | ||||
| 
 | ||||
|     >>> negotiate_locale(['ja', 'en_US'], ['ja_JP', 'en_US']) | ||||
|     'ja_JP' | ||||
| 
 | ||||
|     Some browsers even use an incorrect or outdated language code, such as "no" | ||||
|     for Norwegian, where the correct locale identifier would actually be "nb_NO" | ||||
|     (Bokmål) or "nn_NO" (Nynorsk). The aliases are intended to take care of | ||||
|     such cases, too: | ||||
| 
 | ||||
|     >>> negotiate_locale(['no', 'sv'], ['nb_NO', 'sv_SE']) | ||||
|     'nb_NO' | ||||
| 
 | ||||
|     You can override this default mapping by passing a different `aliases` | ||||
|     dictionary to this function, or you can bypass the behavior althogher by | ||||
|     setting the `aliases` parameter to `None`. | ||||
| 
 | ||||
|     :param preferred: the list of locale strings preferred by the user | ||||
|     :param available: the list of locale strings available | ||||
|     :param sep: character that separates the different parts of the locale | ||||
|                 strings | ||||
|     :param aliases: a dictionary of aliases for locale identifiers | ||||
|     """ | ||||
|     available = [a.lower() for a in available if a] | ||||
|     for locale in preferred: | ||||
|         ll = locale.lower() | ||||
|         if ll in available: | ||||
|             return locale | ||||
|         if aliases: | ||||
|             alias = aliases.get(ll) | ||||
|             if alias: | ||||
|                 alias = alias.replace('_', sep) | ||||
|                 if alias.lower() in available: | ||||
|                     return alias | ||||
|         parts = locale.split(sep) | ||||
|         if len(parts) > 1 and parts[0].lower() in available: | ||||
|             return parts[0] | ||||
|     return None | ||||
| 
 | ||||
| 
 | ||||
| def parse_locale(identifier, sep='_'): | ||||
|     """Parse a locale identifier into a tuple of the form ``(language, | ||||
|     territory, script, variant)``. | ||||
| 
 | ||||
|     >>> parse_locale('zh_CN') | ||||
|     ('zh', 'CN', None, None) | ||||
|     >>> parse_locale('zh_Hans_CN') | ||||
|     ('zh', 'CN', 'Hans', None) | ||||
| 
 | ||||
|     The default component separator is "_", but a different separator can be | ||||
|     specified using the `sep` parameter: | ||||
| 
 | ||||
|     >>> parse_locale('zh-CN', sep='-') | ||||
|     ('zh', 'CN', None, None) | ||||
| 
 | ||||
|     If the identifier cannot be parsed into a locale, a `ValueError` exception | ||||
|     is raised: | ||||
| 
 | ||||
|     >>> parse_locale('not_a_LOCALE_String') | ||||
|     Traceback (most recent call last): | ||||
|       ... | ||||
|     ValueError: 'not_a_LOCALE_String' is not a valid locale identifier | ||||
| 
 | ||||
|     Encoding information and locale modifiers are removed from the identifier: | ||||
| 
 | ||||
|     >>> parse_locale('it_IT@euro') | ||||
|     ('it', 'IT', None, None) | ||||
|     >>> parse_locale('en_US.UTF-8') | ||||
|     ('en', 'US', None, None) | ||||
|     >>> parse_locale('de_DE.iso885915@euro') | ||||
|     ('de', 'DE', None, None) | ||||
| 
 | ||||
|     See :rfc:`4646` for more information. | ||||
| 
 | ||||
|     :param identifier: the locale identifier string | ||||
|     :param sep: character that separates the different components of the locale | ||||
|                 identifier | ||||
|     :raise `ValueError`: if the string does not appear to be a valid locale | ||||
|                          identifier | ||||
|     """ | ||||
|     if '.' in identifier: | ||||
|         # this is probably the charset/encoding, which we don't care about | ||||
|         identifier = identifier.split('.', 1)[0] | ||||
|     if '@' in identifier: | ||||
|         # this is a locale modifier such as @euro, which we don't care about | ||||
|         # either | ||||
|         identifier = identifier.split('@', 1)[0] | ||||
| 
 | ||||
|     parts = identifier.split(sep) | ||||
|     lang = parts.pop(0).lower() | ||||
|     if not lang.isalpha(): | ||||
|         raise ValueError('expected only letters, got %r' % lang) | ||||
| 
 | ||||
|     script = territory = variant = None | ||||
|     if parts: | ||||
|         if len(parts[0]) == 4 and parts[0].isalpha(): | ||||
|             script = parts.pop(0).title() | ||||
| 
 | ||||
|     if parts: | ||||
|         if len(parts[0]) == 2 and parts[0].isalpha(): | ||||
|             territory = parts.pop(0).upper() | ||||
|         elif len(parts[0]) == 3 and parts[0].isdigit(): | ||||
|             territory = parts.pop(0) | ||||
| 
 | ||||
|     if parts: | ||||
|         if len(parts[0]) == 4 and parts[0][0].isdigit() or \ | ||||
|                 len(parts[0]) >= 5 and parts[0][0].isalpha(): | ||||
|             variant = parts.pop() | ||||
| 
 | ||||
|     if parts: | ||||
|         raise ValueError('%r is not a valid locale identifier' % identifier) | ||||
| 
 | ||||
|     return lang, territory, script, variant | ||||
| 
 | ||||
| 
 | ||||
| def get_locale_identifier(tup, sep='_'): | ||||
|     """The reverse of :func:`parse_locale`.  It creates a locale identifier out | ||||
|     of a ``(language, territory, script, variant)`` tuple.  Items can be set to | ||||
|     ``None`` and trailing ``None``\s can also be left out of the tuple. | ||||
| 
 | ||||
|     >>> get_locale_identifier(('de', 'DE', None, '1999')) | ||||
|     'de_DE_1999' | ||||
| 
 | ||||
|     .. versionadded:: 1.0 | ||||
| 
 | ||||
|     :param tup: the tuple as returned by :func:`parse_locale`. | ||||
|     :param sep: the separator for the identifier. | ||||
|     """ | ||||
|     tup = tuple(tup[:4]) | ||||
|     lang, territory, script, variant = tup + (None,) * (4 - len(tup)) | ||||
|     return sep.join(filter(None, (lang, script, territory, variant))) | ||||
							
								
								
									
										1181
									
								
								vendor/babel/dates.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1181
									
								
								vendor/babel/dates.py
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/global.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/global.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										209
									
								
								vendor/babel/localedata.py
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										209
									
								
								vendor/babel/localedata.py
									
									
									
									
										vendored
									
									
								
							|  | @ -1,209 +0,0 @@ | |||
| # -*- coding: utf-8 -*- | ||||
| """ | ||||
|     babel.localedata | ||||
|     ~~~~~~~~~~~~~~~~ | ||||
| 
 | ||||
|     Low-level locale data access. | ||||
| 
 | ||||
|     :note: The `Locale` class, which uses this module under the hood, provides a | ||||
|            more convenient interface for accessing the locale data. | ||||
| 
 | ||||
|     :copyright: (c) 2013 by the Babel Team. | ||||
|     :license: BSD, see LICENSE for more details. | ||||
| """ | ||||
| 
 | ||||
| import os | ||||
| import threading | ||||
| from collections import MutableMapping | ||||
| 
 | ||||
| from babel._compat import pickle | ||||
| 
 | ||||
| 
 | ||||
| _cache = {} | ||||
| _cache_lock = threading.RLock() | ||||
| _dirname = os.path.join(os.path.dirname(__file__), 'localedata') | ||||
| 
 | ||||
| 
 | ||||
| def exists(name): | ||||
|     """Check whether locale data is available for the given locale.  Ther | ||||
|     return value is `True` if it exists, `False` otherwise. | ||||
| 
 | ||||
|     :param name: the locale identifier string | ||||
|     """ | ||||
|     if name in _cache: | ||||
|         return True | ||||
|     return os.path.exists(os.path.join(_dirname, '%s.dat' % name)) | ||||
| 
 | ||||
| 
 | ||||
| def locale_identifiers(): | ||||
|     """Return a list of all locale identifiers for which locale data is | ||||
|     available. | ||||
| 
 | ||||
|     .. versionadded:: 0.8.1 | ||||
| 
 | ||||
|     :return: a list of locale identifiers (strings) | ||||
|     """ | ||||
|     return [stem for stem, extension in [ | ||||
|         os.path.splitext(filename) for filename in os.listdir(_dirname) | ||||
|     ] if extension == '.dat' and stem != 'root'] | ||||
| 
 | ||||
| 
 | ||||
| def load(name, merge_inherited=True): | ||||
|     """Load the locale data for the given locale. | ||||
| 
 | ||||
|     The locale data is a dictionary that contains much of the data defined by | ||||
|     the Common Locale Data Repository (CLDR). This data is stored as a | ||||
|     collection of pickle files inside the ``babel`` package. | ||||
| 
 | ||||
|     >>> d = load('en_US') | ||||
|     >>> d['languages']['sv'] | ||||
|     u'Swedish' | ||||
| 
 | ||||
|     Note that the results are cached, and subsequent requests for the same | ||||
|     locale return the same dictionary: | ||||
| 
 | ||||
|     >>> d1 = load('en_US') | ||||
|     >>> d2 = load('en_US') | ||||
|     >>> d1 is d2 | ||||
|     True | ||||
| 
 | ||||
|     :param name: the locale identifier string (or "root") | ||||
|     :param merge_inherited: whether the inherited data should be merged into | ||||
|                             the data of the requested locale | ||||
|     :raise `IOError`: if no locale data file is found for the given locale | ||||
|                       identifer, or one of the locales it inherits from | ||||
|     """ | ||||
|     _cache_lock.acquire() | ||||
|     try: | ||||
|         data = _cache.get(name) | ||||
|         if not data: | ||||
|             # Load inherited data | ||||
|             if name == 'root' or not merge_inherited: | ||||
|                 data = {} | ||||
|             else: | ||||
|                 parts = name.split('_') | ||||
|                 if len(parts) == 1: | ||||
|                     parent = 'root' | ||||
|                 else: | ||||
|                     parent = '_'.join(parts[:-1]) | ||||
|                 data = load(parent).copy() | ||||
|             filename = os.path.join(_dirname, '%s.dat' % name) | ||||
|             fileobj = open(filename, 'rb') | ||||
|             try: | ||||
|                 if name != 'root' and merge_inherited: | ||||
|                     merge(data, pickle.load(fileobj)) | ||||
|                 else: | ||||
|                     data = pickle.load(fileobj) | ||||
|                 _cache[name] = data | ||||
|             finally: | ||||
|                 fileobj.close() | ||||
|         return data | ||||
|     finally: | ||||
|         _cache_lock.release() | ||||
| 
 | ||||
| 
 | ||||
| def merge(dict1, dict2): | ||||
|     """Merge the data from `dict2` into the `dict1` dictionary, making copies | ||||
|     of nested dictionaries. | ||||
| 
 | ||||
|     >>> d = {1: 'foo', 3: 'baz'} | ||||
|     >>> merge(d, {1: 'Foo', 2: 'Bar'}) | ||||
|     >>> items = d.items(); items.sort(); items | ||||
|     [(1, 'Foo'), (2, 'Bar'), (3, 'baz')] | ||||
| 
 | ||||
|     :param dict1: the dictionary to merge into | ||||
|     :param dict2: the dictionary containing the data that should be merged | ||||
|     """ | ||||
|     for key, val2 in dict2.items(): | ||||
|         if val2 is not None: | ||||
|             val1 = dict1.get(key) | ||||
|             if isinstance(val2, dict): | ||||
|                 if val1 is None: | ||||
|                     val1 = {} | ||||
|                 if isinstance(val1, Alias): | ||||
|                     val1 = (val1, val2) | ||||
|                 elif isinstance(val1, tuple): | ||||
|                     alias, others = val1 | ||||
|                     others = others.copy() | ||||
|                     merge(others, val2) | ||||
|                     val1 = (alias, others) | ||||
|                 else: | ||||
|                     val1 = val1.copy() | ||||
|                     merge(val1, val2) | ||||
|             else: | ||||
|                 val1 = val2 | ||||
|             dict1[key] = val1 | ||||
| 
 | ||||
| 
 | ||||
| class Alias(object): | ||||
|     """Representation of an alias in the locale data. | ||||
| 
 | ||||
|     An alias is a value that refers to some other part of the locale data, | ||||
|     as specified by the `keys`. | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, keys): | ||||
|         self.keys = tuple(keys) | ||||
| 
 | ||||
|     def __repr__(self): | ||||
|         return '<%s %r>' % (type(self).__name__, self.keys) | ||||
| 
 | ||||
|     def resolve(self, data): | ||||
|         """Resolve the alias based on the given data. | ||||
| 
 | ||||
|         This is done recursively, so if one alias resolves to a second alias, | ||||
|         that second alias will also be resolved. | ||||
| 
 | ||||
|         :param data: the locale data | ||||
|         :type data: `dict` | ||||
|         """ | ||||
|         base = data | ||||
|         for key in self.keys: | ||||
|             data = data[key] | ||||
|         if isinstance(data, Alias): | ||||
|             data = data.resolve(base) | ||||
|         elif isinstance(data, tuple): | ||||
|             alias, others = data | ||||
|             data = alias.resolve(base) | ||||
|         return data | ||||
| 
 | ||||
| 
 | ||||
| class LocaleDataDict(MutableMapping): | ||||
|     """Dictionary wrapper that automatically resolves aliases to the actual | ||||
|     values. | ||||
|     """ | ||||
| 
 | ||||
|     def __init__(self, data, base=None): | ||||
|         self._data = data | ||||
|         if base is None: | ||||
|             base = data | ||||
|         self.base = base | ||||
| 
 | ||||
|     def __len__(self): | ||||
|         return len(self._data) | ||||
| 
 | ||||
|     def __iter__(self): | ||||
|         return iter(self._data) | ||||
| 
 | ||||
|     def __getitem__(self, key): | ||||
|         orig = val = self._data[key] | ||||
|         if isinstance(val, Alias): # resolve an alias | ||||
|             val = val.resolve(self.base) | ||||
|         if isinstance(val, tuple): # Merge a partial dict with an alias | ||||
|             alias, others = val | ||||
|             val = alias.resolve(self.base).copy() | ||||
|             merge(val, others) | ||||
|         if type(val) is dict: # Return a nested alias-resolving dict | ||||
|             val = LocaleDataDict(val, base=self.base) | ||||
|         if val is not orig: | ||||
|             self._data[key] = val | ||||
|         return val | ||||
| 
 | ||||
|     def __setitem__(self, key, value): | ||||
|         self._data[key] = value | ||||
| 
 | ||||
|     def __delitem__(self, key): | ||||
|         del self._data[key] | ||||
| 
 | ||||
|     def copy(self): | ||||
|         return LocaleDataDict(self._data.copy(), base=self.base) | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/aa.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/aa.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/aa_DJ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/aa_DJ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/aa_ER.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/aa_ER.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/aa_ET.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/aa_ET.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(Umin_daysqKU
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq }q!Uvariantsq"}q#Ucurrency_namesq$}q%U
unit_patternsq&}q'u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/af.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/af.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/af_NA.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/af_NA.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/af_ZA.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/af_ZA.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(Umin_daysqKU
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq }q!Uvariantsq"}q#Ucurrency_namesq$}q%U
unit_patternsq&}q'u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/agq.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/agq.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/agq_CM.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/agq_CM.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ak.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ak.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ak_GH.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ak_GH.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/am.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/am.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/am_ET.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/am_ET.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(Umin_daysqKU
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq }q!Uvariantsq"}q#Ucurrency_namesq$}q%U
unit_patternsq&}q'u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_001.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_001.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_AE.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_AE.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(U
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}q Uvariantsq!}q"Ucurrency_namesq#}q$U
unit_patternsq%}q&u. | ||||
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_BH.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_BH.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(U
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}q Uvariantsq!}q"Ucurrency_namesq#}q$U
unit_patternsq%}q&u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_DJ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_DJ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_DZ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_DZ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_EG.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_EG.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(Umin_daysqKU
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq }q!Uvariantsq"}q#Ucurrency_namesq$}q%U
unit_patternsq&}q'u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_EH.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_EH.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_ER.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_ER.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_IL.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_IL.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(U
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}q Uvariantsq!}q"Ucurrency_namesq#}q$U
unit_patternsq%}q&u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_IQ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_IQ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_JO.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_JO.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_KM.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_KM.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_KW.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_KW.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(U
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}q Uvariantsq!}q"Ucurrency_namesq#}q$U
unit_patternsq%}q&u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_LB.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_LB.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_LY.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_LY.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_MA.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_MA.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_MR.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_MR.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_OM.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_OM.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(U
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}q Uvariantsq!}q"Ucurrency_namesq#}q$U
unit_patternsq%}q&u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_PS.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_PS.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_QA.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_QA.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_SA.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_SA.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_SD.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_SD.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(Umin_daysqKU
weekend_startqKU	first_dayqKUweekend_endqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq }q!Uvariantsq"}q#Ucurrency_namesq$}q%U
unit_patternsq&}q'u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_SO.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_SO.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_SY.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_SY.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/ar_TD.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/ar_TD.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
Uzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}qUvariantsq}qUcurrency_namesq }q!U
unit_patternsq"}q#u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_TN.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_TN.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_YE.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ar_YE.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/as.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/as.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/as_IN.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/as_IN.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(U
weekend_startqKU	first_dayqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}qUvariantsq }q!Ucurrency_namesq"}q#U
unit_patternsq$}q%u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/asa.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/asa.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/asa_TZ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/asa_TZ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ast.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ast.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/ast_ES.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/ast_ES.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/az.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/az.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Cyrl.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Cyrl.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Cyrl_AZ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Cyrl_AZ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Latn.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Latn.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Latn_AZ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/az_Latn_AZ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bas.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bas.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bas_CM.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bas_CM.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/be.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/be.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/be_BY.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/be_BY.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
U	first_dayqKsUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}qUvariantsq}q Ucurrency_namesq!}q"U
unit_patternsq#}q$u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bem.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bem.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bem_ZM.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bem_ZM.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bez.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bez.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bez_TZ.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bez_TZ.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bg.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bg.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bg_BG.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bg_BG.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bm.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bm.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bm_ML.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bm_ML.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bn.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bn.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/bn_BD.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/bn_BD.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
U	first_dayqKsUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}qUvariantsq}q Ucurrency_namesq!}q"U
unit_patternsq#}q$u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bn_IN.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bn_IN.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bo.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bo.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/bo_CN.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/bo_CN.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
U	first_dayqKsUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}qUvariantsq}q Ucurrency_namesq!}q"U
unit_patternsq#}q$u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bo_IN.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bo_IN.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/br.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/br.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/br_FR.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/br_FR.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/brx.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/brx.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										4
									
								
								vendor/babel/localedata/brx_IN.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								vendor/babel/localedata/brx_IN.dat
									
									
									
									
										vendored
									
									
								
							|  | @ -1,4 +0,0 @@ | |||
| €}q(Ucurrency_symbolsq}qUscientific_formatsq}qUpercent_formatsq}qUnumber_symbolsq}q	Ucurrency_names_pluralq | ||||
| }qU	week_dataq}q
(U
weekend_startqKU	first_dayqKuUzone_formatsq}qUcurrency_formatsq}qU_versionqM5 U	languagesq}qUterritoriesq}U | ||||
| time_zonesq}qUscriptsq}qUdecimal_formatsq}qU | ||||
| meta_zonesq}qUvariantsq }q!Ucurrency_namesq"}q#U
unit_patternsq$}q%u. | ||||
							
								
								
									
										
											BIN
										
									
								
								vendor/babel/localedata/bs.dat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								vendor/babel/localedata/bs.dat
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue
	
	Block a user