I used pdf2text from PDFminer to reduce a PDF to text. Unfortunately it contains special characters. Let me show output from my console
>>>a=pdf_to_text("ap.pdf")
heres a sample of it, a little truncated
>>>a[5000:5500]
'f one architect. Decades ...... but to re\xef\xac\x82ect\none set of design ideas, than to have one that contains many\ngood but independent and uncoordinated ideas.\n1 Joshua Bloch, \xe2\x80\x9cHow to Design a Good API and Why It Matters\xe2\x80\x9d, G......=-3733'
I understood that I must encode it
>>>a[5000:5500].encode('utf-8')
Traceback (most recent call last):
File "<interactive input>", line 1, in <module>
UnicodeDecodeError: 'ascii' codec can't decode byte 0xef in position 237: ordinal not in range(128)
I searched around a bit and tried them, notably Replace special characters in python . The input comes from PDFminer, so its tough (AFAIK) to control that. What is the way to make proper plaintext from this output?
What am I doing wrong?
--A quick fix: change PDFminer's codec to ascii- but it's not a lasting solution--
--Abandoned the quick fix for the answer- changing the codec removes information --
--A relavent topic as mentioned by Maxim http://en.wikipedia.org/wiki/Windows-1251 --
PDFMiner is a text extraction tool for PDF documents. Warning: Starting from version 20191010, PDFMiner supports Python 3 only. For Python 2 support, check out pdfminer.
In Python strings, the backslash "\" is a special character, also called the "escape" character. It is used in representing certain whitespace characters: "\t" is a tab, "\n" is a newline, and "\r" is a carriage return. Conversely, prefixing a special character with "\" turns it into an ordinary character.
This problem often occurs when non-ASCII text is stored in str
objects. What you are trying to do is to encode in utf-8
a string already encoded in some encoding (because it contains characters with codes above 0x7f
).
To encode such a string in utf-8
it has to be first decoded. Assuming that the original text encoding is cp1251
(replace it with your actual encoding), something like the following would do the trick:
u = s.decode('cp1251') # decode from cp1251 byte (str) string to unicode string
s = u.encode('utf-8') # re-encode unicode string to utf-8 byte (str) string
Basically, the above snippet does what iconv --from-code=CP1251 --to-code=UTF-8
command does, i.e. it converts the string from one encoding to another.
Some useful links:
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With