Reader.py :  » Network » Grail-Internet-Browser » grail-0.6 » Python Open Source

Home
Python Open Source
1.3.1.2 Python
2.Ajax
3.Aspect Oriented
4.Blog
5.Build
6.Business Application
7.Chart Report
8.Content Management Systems
9.Cryptographic
10.Database
11.Development
12.Editor
13.Email
14.ERP
15.Game 2D 3D
16.GIS
17.GUI
18.IDE
19.Installer
20.IRC
21.Issue Tracker
22.Language Interface
23.Log
24.Math
25.Media Sound Audio
26.Mobile
27.Network
28.Parser
29.PDF
30.Project Management
31.RSS
32.Search
33.Security
34.Template Engines
35.Test
36.UML
37.USB Serial
38.Web Frameworks
39.Web Server
40.Web Services
41.Web Unit
42.Wiki
43.Windows
44.XML
Python Open Source » Network » Grail Internet Browser 
Grail Internet Browser » grail 0.6 » Reader.py
"""Reader class -- helper to read documents asynchronously."""

import grailutil
import ht_time
import os
import sys
import string
import urlparse
from Tkinter import *
import tktools
from BaseReader import BaseReader
import copy
import regex
import time

# mailcap dictionary
caps = None


# If > 0, profile handle_data and print this many statistics lines
profiling = 0


class ParserWrapper:
    """Provides re-entrance protection around an arbitrary parser object.
    """
    def __init__(self, parser, viewer):
        self.__parser = parser
        self.__viewer = viewer
        self.__pendingdata = ''
        self.__closed = 0
        self.__closing = 0
        self.__level = 0

    def feed(self, data):
        self.__pendingdata = self.__pendingdata + data
        self.__level = self.__level + 1
        if self.__level == 1:
            self.__viewer.unfreeze()
            while self.__pendingdata:
                data = self.__pendingdata
                self.__pendingdata = ''
                self.__parser.feed(data)
            if self.__closing and not self.__closed:
                self.__parser.close()
            self.__viewer.freeze(1)
        self.__level = self.__level - 1

    def close(self):
        self.__closing = 1
        if not self.__level:
            self.__viewer.unfreeze()
            self.__parser.close()
            self.__viewer.freeze()
            self.__closed = 1


class TextLineendWrapper:
    """Perform lineend translation so text data always looks like text (using
    '\n' for lineends)."""

    def __init__(self, parser):
        self.__parser = parser
        self.__last_was_cr = 0

    def feed(self, data):
        if self.__last_was_cr and data[0:1] == '\n':
            data = data[1:]
        self.__last_was_cr = data[-1:] == '\r'
        data = string.replace(data, '\r\n', '\n')
        data = string.replace(data, '\r', '\n')
        self.__parser.feed(data)

    def close(self):
        self.__parser.close()


_hex_digit_values = {
    '0': 0, '1': 1, '2': 2, '3': 3, '4': 4,
    '5': 5, '6': 6, '7': 7, '8': 8, '9': 9,
    'a': 10, 'b': 11, 'c': 12, 'd': 13, 'e': 14, 'f': 15,
    'A': 10, 'B': 11, 'C': 12, 'D': 13, 'E': 14, 'F': 15,
    }

class QuotedPrintableWrapper:
    """Wrap a parser object with a quoted-printable decoder.  Conforms to
    parser protocol."""
    def __init__(self, parser):
        """Initialize the decoder.  Pass in the real parser as a parameter."""
        self.__parser = parser
        self.__buffer = ''
        self.__last_was_cr = 0

    def feed(self, data):
        """Decode data and feed as much as possible to the real parser."""
        # handle lineend translation inline
        if self.__last_was_cr and data[0:1] == '\n':
            data = data[1:]
        self.__last_was_cr = data[-1:] == '\r'
        data = string.replace(data, '\r\n', '\n')
        data = string.replace(data, '\r', '\n')

        # now get the real buffer
        data = self.__buffer + data
        pos = string.find(data, '=')
        if pos == -1:
            self.__parser.feed(data)
            self.__buffer = ''
            return
        s = data[:pos]
        data = data[pos:]
        pos = 0
        while pos <= len(data):
            xx = data[:2]
            if xx == '=\n':
                data = data[2:]
            elif xx == '==':
                s = s + '='
                data = data[2:]
            elif len(data) >= 3:
                try:
                    v = _hex_digit_values[data[1]] << 4
                    v = v | _hex_digit_values[data[2]]
                except KeyError:
                    s = s + '='
                    data = data[1:]
                    print "invalid quoted-printable encoding -- skipping '='"
                else:
                    s = s + chr(v)
                    data = data[3:]
            elif len(data) < 3:
                # wait for more data
                break
            else:
                s = s + '='
                data = data[1:]
                print "invalid quoted-printable encoding -- skipping '='"
            # now look for the next '=':
            pos = string.find(data, '=')
            if pos == -1:
                s = s + data
                data = ''
            else:
                s = s + data[:pos]
                data = data[pos:]
                pos = 0
        self.__parser.feed(s)
        self.__buffer = data

    def close(self):
        """Flush any remaining encoded data and feed it to the parser; there's
        no way to properly decode it.  Close the parser afterwards."""
        # just ignore that we couldn't parse it!
        self.__parser.feed(self.__buffer)
        self.__parser.close()


class Base64Wrapper:
    """Decode base64-encoded data on the fly, and pass it on to the real
    type-specific parser."""

    def __init__(self, parser):
        self.__parser = parser
        self.__buffer = ''
        self.__app = parser.viewer.context.app
        self.__last_was_cr = 0

    def feed(self, data):
        # handle lineend translation inline
        if self.__last_was_cr and data[0:1] == '\n':
            data = data[1:]
        self.__last_was_cr = data[-1:] == '\r'
        data = string.replace(data, '\r\n', '\n')
        data = string.replace(data, '\r', '\n')

        # now get the real buffer
        data = self.__buffer + data
        lines = string.split(data, '\n')
        if len(lines) > 1:
            data = lines[-1]
            del lines[-1]
            stuff = ''
            # do it this way to handle as much of the data as possible
            # before barfing on it
            while lines:
                try:
                    bin = binascii.a2b_base64(lines[0])
                except (binascii.Error, binascii.Incomplete):
                    self.__app.exception_dialog("while decoding base64 data")
                else:
                    stuff = stuff + bin
                del lines[0]
            lines.append(data)
            data = string.join(lines, '\n')
            if stuff:
                self.__parser.feed(stuff)
        self.__buffer = data

    def close(self):
        if self.__buffer:
            try:
                bin = binascii.a2b_base64(self.__buffer)
            except (binascii.Error, binascii.Incomplete):
                # can't do anything with it.... toss a warning?
                pass
            else:
                self.__parser.feed(bin)
        self.__parser.close()


class DeflateWrapper:
    """Decompress deflated data incrementally and pass is on to the real
    type-specific handler."""

    def __init__(self, parser):
        import zlib
        self.__parser = parser
        self.__decompressor = zlib.decompressobj(-zlib.MAX_WBITS)

    def feed(self, data):
        data = self.__decompressor.decompress(data)
        self.__parser.feed(data)

    def close(self):
        data = self.__decompressor.flush()
        if data:
            self.__parser.feed(data)
        self.__parser.close()


class GzipWrapper:
    """Decompress gzipped data incrementally and pass it on to the real
    type-specific handler."""

    BASE_HEADER_LENGTH = 10

    def __init__(self, parser):
        self.__parser = parser
        self.__header = 0
        self.__fextra = 0
        self.__fname = 0
        self.__fcomment = 0
        self.__fhcrc = 0
        self.__in_data = 0
        self.__buffer = ''

    def feed(self, data):
        if not self.__in_data:
            data = self.__feed_header(data)
        if self.__in_data and data:
            data = self.__decompressor.decompress(data)
            if data:
                self.__parser.feed(data)

    def __feed_header(self, data):
        data = self.__buffer + data
        if not self.__header:
            if len(data) >= self.BASE_HEADER_LENGTH:
                if data[:3] != '\037\213\010':
                    raise RuntimeError, "invalid gzip header"
                self.__flag = ord(data[3])
                self.__header = 1
                data = data[10:]
        if self.__header:
            ok = 1
            if not self.__fextra:
                data, ok = self.__read_fextra(data)
            if ok and not self.__fname:
                data, ok = self.__read_fname(data)
            if ok and not self.__fcomment:
                data, ok = self.__read_fcomment(data)
            if ok and not self.__fhcrc:
                data, ok = self.__read_fhcrc(data)
            if ok:
                self.__buffer = ''
                self.__in_data = 1
                # Call the constructor exactly this way to get gzip-style
                # compression.  Omitting the optional arg doesn't lead to
                # gzip-compatible decompression.
                decompressor = zlib.decompressobj(-zlib.MAX_WBITS)
                self.__decompressor = decompressor
            else:
                self.__buffer = data
        return data

    def __read_fextra(self, data):
        if not self.__flag & gzip.FEXTRA:
            self.__fextra = 1
            return data, 1
        if len(data) < 2:
            return data, 0
        length = ord(data[0]) + (256 * ord(data[1]))
        if len(data) < (length + 2):
            return data, 0
        self.__fextra = 1
        self.extra = data[2:length]
        return data[length + 2:], 1

    def __read_fname(self, data):
        if self.__flag & gzip.FNAME:
            data, ok, stuff = self.__read_zstring(data)
        else:
            ok, stuff = 1, None
        if ok:
            self.__fname = 1
            self.name = stuff
        return data, ok

    def __read_fcomment(self, data):
        if self.__flag & gzip.FCOMMENT:
            data, ok, stuff = self.__read_zstring(data)
        else:
            ok, stuff = 1, None
        if ok:
            self.__fcomment = 1
            self.comment = stuff
        return data, ok

    def __read_fhcrc(self, data):
        if self.__flag & gzip.FHCRC:
            if len(data) >= 2:
                self.__fhcrc = 1
                data = data[2:]
        else:
            self.__fhcrc = 1
        return data, self.__fhcrc

    def __read_zstring(self, data):
        """Attempt to read a null-terminated string."""
        if '\0' in data:
            stuff = data[:string.index(data, '\0')]
            return data[len(stuff) + 1:], 1, stuff
        return data, 0, None

    def close(self):
        if self.__in_data:
            data = self.__decompressor.flush()
            if data:
                self.__parser.feed(data)
        self.__parser.close()


# This table maps content-transfer-encoding values to the appropriate
# decoding wrappers.  It should not be needed with HTTP (1.1 explicitly
# forbids it), but it's never a good idea to ignore the possibility.
#
transfer_decoding_wrappers = {
    "quoted-printable": QuotedPrintableWrapper,
    }

try:
    import binascii
except ImportError:
    pass
else:
    transfer_decoding_wrappers["base64"] = Base64Wrapper


# This table maps content-encoding values to the appropriate decoding
# wrappers.  This can (and should if you care about bandwidth) be used
# whenever possible.  We need to send an accept-encoding header when
# push comes to shove, but we'll ignore that for the moment.
#
content_decoding_wrappers = {}
try:
    import zlib
    import gzip
except ImportError, error:
    pass
else:
    content_decoding_wrappers["deflate"] = DeflateWrapper
    content_decoding_wrappers["gzip"] = GzipWrapper
    content_decoding_wrappers["x-gzip"] = GzipWrapper


def get_encodings(headers):
    content_encoding = transfer_encoding = None
    if headers.has_key("content-encoding"):
        content_encoding = string.lower(headers["content-encoding"])
    if headers.has_key("content-transfer-encoding"):
        transfer_encoding = string.lower(headers["content-transfer-encoding"])
    return content_encoding, transfer_encoding


def wrap_parser(parser, ctype, content_encoding=None, transfer_encoding=None):
    if ctype[:5] == "text/":
        parser = TextLineendWrapper(parser)
    if content_encoding:
        parser = content_decoding_wrappers[content_encoding](parser)
    if transfer_encoding:
        parser = transfer_decoding_wrappers[transfer_encoding](parser)
    return parser
    

def get_content_encodings():
    """Return a list of supported content-encoding values."""
    return content_decoding_wrappers.keys()


def get_transfer_encodings():
    """Return a list of supported content-transfer-encoding values."""
    return transfer_decoding_wrappers.keys()


def support_encodings(content_encoding, transfer_encoding):
    """Return true iff both content and content-transfer encodings are
    supported."""
    if content_encoding \
       and not content_decoding_wrappers.has_key(content_encoding):
        return 0
    if transfer_encoding \
       and not transfer_decoding_wrappers.has_key(transfer_encoding):
        return 0
    return 1


class Reader(BaseReader):

    """Helper class to read documents asynchronously.

    This is created by the Context.load() method and it is deleted
    when the document is fully read or when the user stops it.

    There should never be two Reader instances attached to the same
    Context instance, but if there were, the only harm done would be
    that their output would be merged in the context's viewer.

    """

    def __init__(self, context, url, method, params, show_source, reload,
                 data=None, scrollpos=None):

        self.last_context = context
        self.method = method
        self.params = copy.copy(params)
        self.show_source = show_source
        self.reload = reload
        self.data = data
        self.scrollpos = scrollpos

        self.save_file = None
        self.save_mailcap = None
        self.user_passwd = None
        self.maxrestarts = 10
        self.url = ''

        if url: self.restart(url)

    def restart(self, url):
        self.maxrestarts = self.maxrestarts - 1

        self.viewer = self.last_context.viewer
        self.app = self.last_context.app

        self.parser = None

        tuple = urlparse.urlparse(url)
        # it's possible that the url send in a 301 or 302 error is a
        # relative URL.  if there's no scheme or netloc in the
        # returned tuple, try joining the URL with the previous URL
        # and retry parsing it.
        if not (tuple[0] and tuple[1]):
            url = urlparse.urljoin(self.url, url)
            tuple = urlparse.urlparse(url)
        self.url = url

        self.fragment = tuple[-1]
        tuple = tuple[:-1] + ("",)
        if self.user_passwd:
            netloc = tuple[1]
            i = string.find(netloc, '@')
            if i >= 0: netloc = netloc[i+1:]
            netloc = self.user_passwd + '@' + netloc
            tuple = (tuple[0], netloc) + tuple[2:]
        realurl = urlparse.urlunparse(tuple)

        # Check first to see if the previous Context has any protocol handlers
        api = self.last_context.get_local_api(realurl, self.method,
                                              self.params)
        if not api:
            if self.app:
                api = self.app.open_url(realurl,
                                        self.method, self.params, self.reload,
                                        data=self.data)
            else:
                import protocols
                api = protocols.protocol_access(realurl,
                                                self.method, self.params,
                                                data=self.data)

        BaseReader.__init__(self, self.last_context, api)

    def stop(self):
        BaseReader.stop(self)
        if self.parser:
            parser = self.parser
            self.parser = None
            parser.close()

    def handle_error(self, errcode, errmsg, headers):
        if self.save_file:
            self.save_file.close()
            self.save_file = None
            if self.save_mailcap:
                try:
                    os.unlink(self.save_filename)
                except os.error:
                    pass
        BaseReader.handle_error(self, errcode, errmsg, headers)

    def handle_meta_prelim(self, errcode, errmsg, headers):
        self.last_context.set_headers(headers)
        if self.save_file:
            if errcode != 200:
                self.stop()
                self.handle_error(errcode, errmsg, headers)
            return

        if errcode == 204:
            self.last_context.viewer.remove_temp_tag(histify=1)
            self.app.global_history.remember_url(self.url)
            self.stop()
            return

        if errcode in (301, 302) and headers.has_key('location'):
            url = headers['location']
            if self.maxrestarts > 0:
                # remember the original click location
                self.app.global_history.remember_url(self.url)
                self.stop()
                # Always do a "GET" on the redirected URL
                self.method = 'GET'
                self.data = ""
                self.restart(url)
                return

        if errcode == 401:
            if self.handle_auth_error(errcode, errmsg, headers):
                return

        return 1

    def handle_meta(self, errcode, errmsg, headers):
        if not self.handle_meta_prelim(errcode, errmsg, headers):
            return

        # This updates the attributes in the bookmarks database.
        try:
            bkmks = self.last_context.app.bookmarks_controller
        except AttributeError:
            pass
        else:
            last_modified = None
            if headers.has_key("last-modified"):
                try: last_modified = ht_time.parse(headers["last-modified"])
                except ValueError: pass
            bkmks.record_visit(self.url, last_modified)

        content_encoding, transfer_encoding = get_encodings(headers)
        if headers.has_key('content-type'):
            content_type = headers['content-type']
            if ';' in content_type:
                content_type = string.strip(
                    content_type[:string.index(content_type, ';')])
        else:
            content_type, encoding = self.app.guess_type(self.url)
            if not content_encoding:
                content_encoding = encoding
        real_content_type = content_type or "unknown"
        real_content_encoding = content_encoding
        if (transfer_encoding
            and not transfer_decoding_wrappers.has_key(transfer_encoding)) \
            or (content_encoding
                and not content_decoding_wrappers.has_key(content_encoding)):
            # XXX provisional hack -- change content type to octet stream
            content_type = "application/octet-stream"
            transfer_encoding = None
            content_encoding = None
        if not content_type:
            content_type = "text/plain" # Last resort guess only

        istext = content_type and content_type[:5] == 'text/' \
                 and not (content_encoding or transfer_encoding)
        if self.show_source and istext:
            content_type = 'text/plain'
        parserclass = self.find_parser_extension(content_type)
        if not parserclass and istext:
            if content_type != 'text/plain':
                # still need to check for text/plain
                parserclass = self.find_parser_extension('text/plain')
            if not parserclass:
                parserclass = TextParser

        if not parserclass:
            # Don't know how to display this.
            # First consult mailcap.
            import mailcap
            global caps
            if not caps:
                caps = mailcap.getcaps()
            if caps:
                plist = [] # XXX Should be taken from Content-type header
                command, entry = mailcap.findmatch(
                    caps, content_type, 'view', "/dev/null", plist)
                if command:
                    # Retrieve to temporary file.
                    import tempfile
                    self.save_mailcap = command
                    self.save_filename = tempfile.mktemp()
                    self.save_content_type = content_type
                    self.save_plist = plist
                    self.save_file = open(self.save_filename, "wb")
                    # remember the original click location
                    self.app.global_history.remember_url(self.url)
                    self.viewer.remove_temp_tag(histify=1)
                    return
            # No relief from mailcap either.
            # Ask the user whether and where to save it.
            # Stop the transfer, and restart when we're ready.
            context = self.last_context
            # TBD: hack so that Context.rmreader() doesn't call
            # Viewer.remove_temp_tag().  We'll call that later
            # explicitly once we know whether the file has been saved
            # or not.
            context.source = None
            self.stop()
            context.message("Wait for save dialog...")
            encoding = ''
            if real_content_encoding:
                encoding = real_content_encoding + "ed "
                if encoding[:2] == "x-":
                    encoding = encoding[2:]
            encoding_label = "MIME type: %s%s" % (encoding, real_content_type)
            import FileDialog
            fd = FileDialog.SaveFileDialog(context.root)
            label = Label(fd.top, text=encoding_label)
            label.pack(before=fd.filter)
            # give it a default filename on which save within the
            # current directory
            urlasfile = string.splitfields(self.url, '/')
            fn = fd.go(default=urlasfile[-1], key="save")
            if not fn:
                # User canceled.  Stop the transfer.
                self.viewer.remove_temp_tag()
                return
            self.viewer.remove_temp_tag(histify=1)
            self.app.global_history.remember_url(self.url)
            # Prepare to save.
            # Always save in binary mode.
            try:
                self.save_file = open(fn, "wb")
            except IOError, msg:
                context.error_dialog(IOError, msg)
                return
            TransferDisplay(context, fn, self)
            return

        if headers.has_key('window-target'):
            target = headers['window-target']
            if target:
                context = self.context.find_window_target(target)
                if context is not self.context:
                    self.context.rmreader(self)
                    self.context = self.last_context = context
                    self.context.addreader(self)
                    self.viewer = self.context.viewer
        self.context.clear_reset()
        self.context.set_headers(headers)
        self.context.set_url(self.url)
        parser = parserclass(self.viewer, reload=self.reload)
        # decode the content
        parser = wrap_parser(parser, content_type,
                             content_encoding, transfer_encoding)
        # protect from re-entrance
        self.parser = ParserWrapper(parser, self.viewer)


    def handle_auth_error(self, errcode, errmsg, headers):
        # Return nonzero if handle_error() should return now
        if not headers.has_key('www-authenticate') \
           or self.maxrestarts <= 0:
            return

        cred_headers = {}
        for k in headers.keys():
            cred_headers[string.lower(k)] = headers[k]
        cred_headers['request-uri'] = self.url

        if self.params.has_key('Authorization'):
            self.app.auth.invalidate_credentials(cred_headers,
                                                 self.params['Authorization'])
            return

        self.stop()
        credentials = self.app.auth.request_credentials(cred_headers)
        if credentials.has_key('Authorization'):
            for k,v in credentials.items():
                self.params[k] = v
            self.restart(self.url)
            return 1
        # couldn't figure out scheme
        self.maxrestarts = 0
        self.restart(self.url)
        return

    def handle_data(self, data):
        if self.save_file:
            self.save_file.write(data)
            return
        try:
            self.parser.feed(data)
        except IOError, msg:
            self.stop()
            try:
                errno, errmsg = msg
            except:
                errno, errmsg = 0, str(msg)
            self.handle_error(errno, errmsg, [])

    if profiling:
        bufsize = 8*1024
        _handle_data = handle_data
        def handle_data(self, data):
            n = profiling
            import profile, pstats
            prof = profile.Profile()
            prof.runcall(self._handle_data, data)
            stats = pstats.Stats(prof)
            stats.strip_dirs().sort_stats('time').print_stats(n)
            stats.strip_dirs().sort_stats('cum').print_stats(n)

    def handle_eof(self):
        if not self.save_file:
            if self.fragment:
                self.viewer.scroll_to(self.fragment)
            elif self.scrollpos:
                self.viewer.scroll_to_position(self.scrollpos)
            return
        self.save_file.close()
        self.save_file = None
        if not self.save_mailcap:
            self.last_context.message("Saved.")
            return
        import mailcap
        command, entry = mailcap.findmatch(
            caps, self.save_content_type, 'view',
            self.save_filename, self.save_plist)
        if not command:
            command = self.save_mailcap
        self.last_context.message("Mailcap: %s" % command)
        command = "(%s; rm -f %s)&" % (command, self.save_filename)
        sts = os.system(command)
        if sts:
            print "Exit status", sts, "from command", command

    def find_parser_extension(self, content_type):
        app = self.context.app
        ext = app.find_type_extension('filetypes', content_type)
        if ext:
            return ext.parse
        return None


class LoginDialog:

    def __init__(self, master, netloc, realmvalue):
        self.root = tktools.make_toplevel(master,
                                          title="Authentication Dialog")
        self.prompt = Label(self.root,
                            text="Enter user authentication\nfor %s on %s" %
                            (realmvalue, netloc))
        self.prompt.pack(side=TOP)
        self.user_entry, dummy = tktools.make_form_entry(self.root, "User:")
        self.user_entry.focus_set()
        self.user_entry.bind('<Return>', self.user_return_event)
        self.passwd_entry, dummy = \
                           tktools.make_form_entry(self.root, "Password:")
        self.passwd_entry.config(show="*")
        self.passwd_entry.bind('<Return>', self.ok_command)
        self.ok_button = Button(self.root, text="OK", command=self.ok_command)
        self.ok_button.pack(side=LEFT)
        self.cancel_button = Button(self.root, text="Cancel",
                                    command=self.cancel_command)
        self.cancel_button.pack(side=RIGHT)

        self.user_passwd = None

        tktools.set_transient(self.root, master)

        self.root.grab_set()

    def go(self):
        try:
            self.root.mainloop()
        except SystemExit:
            return self.user_passwd

    def user_return_event(self, event):
        self.passwd_entry.focus_set()

    def ok_command(self, event=None):
        user = string.strip(self.user_entry.get())
        passwd = string.strip(self.passwd_entry.get())
        if not user:
            self.root.bell()
            return
        self.user_passwd = user + ':' + passwd
        self.goaway()

    def cancel_command(self):
        self.user_passwd = None
        self.goaway()

    def goaway(self):
        self.root.destroy()
        raise SystemExit


from formatter import AS_IS


class TextParser:

    title = ""

    def __init__(self, viewer, reload=0):
        self.viewer = viewer
        self.viewer.new_font((AS_IS, AS_IS, AS_IS, 1))

    def feed(self, data):
        self.viewer.send_literal_data(data)

    def close(self):
        pass


# This constant is the minimum interval between the times we force the
# display to be updated during an asynchronous download.  This makes the
# display update less "choppy" over fast links, where the display might
# not get updated because another socket event occurs before re-entering
# the main loop.  See TransferDisplay.write() for use.
#
TRANSFER_STATUS_UPDATE_PERIOD = 0.5

DARK_BLUE = "#00008b"
LIGHT_BLUE = "#b0e0e6"


class TransferDisplay:
    """A combined browser / viewer for asynchronous file transfers."""

    def __init__(self, old_context, filename, reader, restart=1):
        url = old_context.get_url()
        headers = old_context.get_headers()
        self.app = old_context.browser.app
        self.root = tktools.make_toplevel(
            old_context.browser.master, class_="GrailTransfer")
        self.root.protocol("WM_DELETE_WINDOW", self.stop)
        import Context
        self.context = Context.SimpleContext(self, self)
        self.context._url = self.context._baseurl = url
        reader.last_context = self.context
        self.__filename = filename
        self.__reader = reader
        self.__save_file = reader.save_file
        reader.save_file = self
        if filename:
            self.root.title("Grail: Downloading "
                           + os.path.basename(filename))
        else:
            self.root.title("Grail Download")
        self.root.iconname("Download")
        #
        self.content_length = None
        if headers.has_key('content-length'):
            self.content_length = string.atoi(headers['content-length'])
        self.create_widgets(url, filename, self.content_length)
        #
        if restart:
            reader.restart(reader.url)
        reader.bufsize = 8096
        tktools.set_transient(self.root, old_context.browser.master)
        history = old_context.app.global_history
        if not history.inhistory_p(url):
            history.remember_url(url)
        self.root.update_idletasks()

    def create_widgets(self, url, filename, content_length):
        """Create the widgets in the Toplevel instance."""
        fr, topfr, botfr = tktools.make_double_frame(self.root)
        Label(topfr, text="Downloading %s" % os.path.basename(filename)
              ).pack(anchor=W, pady='1m')
        Frame(topfr, borderwidth=1, height=2, relief=SUNKEN
              ).pack(fill=X, pady='1m')
        self.make_labeled_field(topfr, "Source:", url)['width'] = 45
        self.make_labeled_field(topfr, "Destination:", filename)
        Button(botfr, command=self.stop, text="Stop").pack()
        if content_length:
            self.make_progress_bar(content_length, topfr)
        frame = Frame(topfr)
        frame.pack(fill=X)
        self.__bytes = self.make_labeled_field(frame, "Bytes:", "0", LEFT)
        if content_length:
            self.__bytes['width'] = len(`content_length`) + 2
            self.__percent = self.make_labeled_field(
                frame, "Complete:", self.__bytespat % 0.0, LEFT)
        else:
            self.__percent = None

    __boldpat = regex.compile("-\([a-z]*bold\)-", regex.casefold)
    __datafont = None
    def make_labeled_field(self, master, labeltext, valuetext='', side=TOP):
        frame = Frame(master)
        frame.pack(pady='1m', side=side, anchor=W)
        label = Label(frame, anchor=E, text=labeltext, width=10)
        label.pack(side=LEFT)
        value = Label(frame, anchor=W, text=valuetext)
        if self.__datafont is None:
            # try to get a medium-weight version of the font if bold:
            font = label['font']
            pos = self.__boldpat.search(font) + 1
            if pos:
                end = pos + len(self.__boldpat.group(1))
                self.__datafont = "%smedium%s" % (font[:pos], font[end:])
        if self.__datafont:
            try: value['font'] = self.__datafont
            except TclError: self.__datafont = ''
        value.pack(side=RIGHT, fill=X, expand=1)
        return value

    def message(self, string):
        pass

    __progbar = None
    __bytespat = "%.1f%%"
    def make_progress_bar(self, size, frame):
        self.__bytespat = "%.1f%% of " + grailutil.nicebytes(size)
        self.__maxsize = 1.0 * size     # make it a float for future calc.
        f = Frame(frame, relief=SUNKEN, borderwidth=1, background=LIGHT_BLUE,
                  height=20, width=202)
        f.pack(pady='1m')

        self.__progbar = Frame(f, width=1, background=DARK_BLUE,
                               height=string.atoi(f.cget('height'))
                               - 2*string.atoi(f.cget('borderwidth')))
        self.__progbar.place(x=0, y=0)

    def stop(self):
        self.close()
        if os.path.isfile(self.__filename):
            try: os.unlink(self.__filename)
            except IOError, msg: self.context.error_dialog(IOError, msg)

    # file-like methods; these allow us to intercept the close() method
    # on the reader's save file object

    __datasize = 0
    __prevtime = 0.0
    def write(self, data):
        self.__save_file.write(data)
        datasize = self.__datasize = self.__datasize + len(data)
        self.__bytes['text'] = datasize
        if self.__progbar:
            self.__progbar.config(
                width=max(1, int(datasize * (200 / self.__maxsize))))
            self.__percent['text'] = (
                self.__bytespat % (100.0 * (datasize / self.__maxsize)))
            t = time.time()
            if t - self.__prevtime >= TRANSFER_STATUS_UPDATE_PERIOD:
                self.root.update_idletasks()
                self.__prevtime = t

    def close(self):
        # make sure the 100% mark is updated on the display:
        self.root.update_idletasks()
        self.__reader.stop()
        self.__save_file.close()
        self.__reader.save_file = self.__save_file
        self.__save_file = self.__reader = None
        self.root.destroy()
www.java2java.com | Contact Us
Copyright 2009 - 12 Demo Source and Support. All rights reserved.
All other trademarks are property of their respective owners.