Вот несколько идей:
(0) Объясните «файл» и «иногда»: вы действительно имеете в виду, что он иногда работает и иногда дает сбой с тем же файлом ?
Выполните следующие действия для каждого файла с ошибками:
(1) Узнайте, что находится в файле на момент, когда он жалуется:
text = open("the_file.xml", "rb").read()
err_col = 52459
print repr(text[err_col-50:err_col+100]) # should include the error text
print repr(text[:50]) # show the XML declaration
(2) Бросьте свой файл в веб-службу проверки XML, например, http://www.validome.org/xml/ или http://validator.aborla.net/
и отредактируйте свой вопрос, чтобы отобразить результаты.
Обновление : вот минимальный xml-файл, иллюстрирующий вашу проблему:
[badcharref.xml]
<a></a>
[Python 2.7.1 output]
>>> import xml.etree.ElementTree as ET
>>> it = ET.iterparse(file("badcharref.xml"))
>>> for ev, el in it:
... print el.tag
...
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "C:\python27\lib\xml\etree\ElementTree.py", line 1258, in next
self._parser.feed(data)
File "C:\python27\lib\xml\etree\ElementTree.py", line 1624, in feed
self._raiseerror(v)
File "C:\python27\lib\xml\etree\ElementTree.py", line 1488, in _raiseerror
raise err
xml.etree.ElementTree.ParseError: reference to invalid character number: line 1, column 3
>>>
Не все допустимые символы Юникода допустимы в XML. См. XML 1.0 Спецификация .
Возможно, вы захотите проверить свои файлы с помощью регулярных выражений, таких как r'&#([0-9]+);'
и r'&#x([0-9A-Fa-f]+);'
, преобразовать сопоставленный текст в порядковый номер и проверить действительный список из спецификации, т.е. #x9 | #xA | #xD | [#x20-#xD7FF] | [#xE000-#xFFFD] | [#x10000-#x10FFFF]
... или, возможно, синтаксически недопустимая ссылка на числовые символы, например, не заканчивается ;
', &#not-a-digit
и т. д.
Обновление 2 Я ошибся, число в сообщении об ошибке ElementTree считает кодовые точки Unicode, а не байты. См. Приведенный ниже код и фрагменты из выходных данных, запустив его над двумя неверными файлами.
# coding: ascii
# Find numeric character references that refer to Unicode code points
# that are not valid in XML.
# Get byte offsets for seeking etc in undecoded file bytestreams.
# Get unicode offsets for checking against ElementTree error message,
# **IF** your input file is small enough.
BYTE_OFFSETS = True
import sys, re, codecs
fname = sys.argv[1]
print fname
if BYTE_OFFSETS:
text = open(fname, "rb").read()
else:
# Assumes file is encoded in UTF-8.
text = codecs.open(fname, "rb", "utf8").read()
rx = re.compile("&#([0-9]+);|&#x([0-9a-fA-F]+);")
endpos = len(text)
pos = 0
while pos < endpos:
m = rx.search(text, pos)
if not m: break
mstart, mend = m.span()
target = m.group(1)
if target:
num = int(target)
else:
num = int(m.group(2), 16)
# #x9 | #xA | #xD | [#x20-#xD7FF] | [#xE000-#xFFFD] | [#x10000-#x10FFFF]
if not(num in (0x9, 0xA, 0xD) or 0x20 <= num <= 0xD7FF
or 0xE000 <= num <= 0xFFFD or 0x10000 <= num <= 0x10FFFF):
print mstart, m.group()
pos = mend
Выход:
comments.xml
6615405 
10205764 �
10213901 �
10213936 �
10214123 �
13292514 
...
155656543 
155656564 
157344876 
157722583 
posts.xml
7607143 
12982273 
12982282 
12982292 
12982302 
12982310 
16085949 
16085955 
...
36303479 
36303494  <<=== whoops
38942863 
...
785292911 
801282472 
848911592