python-tldp/tldp/typeguesser.py

126 lines
3.9 KiB
Python
Raw Normal View History

2016-02-11 03:22:23 +00:00
#! /usr/bin/python
2016-02-18 21:25:02 +00:00
# -*- coding: utf8 -*-
2016-04-29 15:02:02 +00:00
#
# Copyright (c) 2016 Linux Documentation Project
2016-02-11 03:22:23 +00:00
from __future__ import absolute_import, division, print_function
2016-02-11 03:22:23 +00:00
import os
import codecs
2016-02-11 03:22:23 +00:00
import inspect
import logging
2016-02-11 03:22:23 +00:00
2016-02-25 17:51:17 +00:00
import tldp.doctypes
2016-02-11 03:22:23 +00:00
2016-03-01 04:33:14 +00:00
logger = logging.getLogger(__name__)
2016-02-25 17:51:17 +00:00
def getDoctypeMembers(membertype):
'''returns a list of tldp.doctypes; convenience function'''
found = list()
for name, member in inspect.getmembers(tldp.doctypes, membertype):
logger.debug("Located %s %s (%r).", membertype.__name__, name, member)
found.append(member)
return found
def getDoctypeClasses():
'''returns a list of the classes known in tldp.doctypes
2016-02-19 07:07:44 +00:00
This is the canonical list of doctypes which are recognized and capable of
being processed into outputs. See tldp.doctypes for more information.
'''
2016-02-25 17:51:17 +00:00
return getDoctypeMembers(inspect.isclass)
2016-02-11 03:22:23 +00:00
def guess(fname):
2016-02-19 07:07:44 +00:00
'''return a tldp.doctype class which is a best guess for document type
:parama fname: A filename.
2016-02-19 07:07:44 +00:00
The guess function will try to guess the document type (doctype) from the
file extension. If extension matching produces multiple possible doctype
matches (e.g. .xml or .sgml), the guess function will then use signature
matching to find the earliest match in the file for a signature.
If there are multiple signature matches, it will choose the signature
matching at the earliest position in the file.
Bugs/shortcomings:
* This is only a guesser.
* When signature matching, it reports first signature it discovers in
any input file.
* It could/should read more than 1024 bytes (cf. SignatureChecker)
especially if it cannot return any result.
* It could/should use heuristics or something richer than signatures.
'''
2016-02-11 03:22:23 +00:00
try:
stem, ext = os.path.splitext(fname)
except (AttributeError, TypeError):
2016-02-11 03:22:23 +00:00
return None
if not ext:
logger.debug("%s no file extension, skipping %s.", stem, ext)
return None
possible = [t for t in knowndoctypes if ext in t.extensions]
logger.debug("Possible: %r", possible)
2016-02-11 03:22:23 +00:00
if not possible:
logger.debug("%s unknown extension %s.", stem, ext)
2016-02-11 03:22:23 +00:00
return None
2016-02-11 03:22:23 +00:00
if len(possible) == 1:
doctype = possible.pop()
return doctype
# -- for this extension, multiple document types, probably SGML, XML
#
logger.debug("%s multiple possible doctypes for extension %s on file %s.",
stem, ext, fname)
2016-02-11 03:22:23 +00:00
for doctype in possible:
logger.debug("%s extension %s could be %s.", stem, ext, doctype)
2016-02-11 03:22:23 +00:00
try:
with codecs.open(fname, encoding='utf-8') as f:
buf = f.read(1024)
except UnicodeDecodeError:
# -- a wee bit ugly, but many SGML docs used iso-8859-1, so fall back
with codecs.open(fname, encoding='iso-8859-1') as f:
buf = f.read(1024)
2016-02-11 03:22:23 +00:00
guesses = list()
for doctype in possible:
sindex = doctype.signatureLocation(buf, fname)
2016-02-11 03:22:23 +00:00
if sindex is not None:
guesses.append((sindex, doctype))
if not guesses:
logger.warning("%s no matching signature found for %s.",
stem, fname)
2016-02-11 03:22:23 +00:00
return None
if len(guesses) == 1:
_, doctype = guesses.pop()
return doctype
# -- OK, this is unusual; we still found multiple document type
# signatures. Seems rare but unlikely, so we should choose the
# first signature in the file as the more likely document type.
#
guesses.sort()
logger.info("%s multiple doctype guesses for file %s", stem, fname)
2016-02-11 03:22:23 +00:00
for sindex, doctype in guesses:
logger.info("%s could be %s (sig at pos %s)", stem, doctype, sindex)
logger.info("%s going to guess %s for %s", stem, doctype, fname)
_, doctype = guesses.pop(0)
2016-02-11 03:22:23 +00:00
return doctype
knowndoctypes = getDoctypeClasses()
2016-02-11 03:22:23 +00:00
knownextensions = set()
for x in knowndoctypes:
2016-02-11 03:22:23 +00:00
knownextensions.update(x.extensions)
#
# -- end of file