aboutsummaryrefslogblamecommitdiffstats
path: root/embed/mozilla/EphyAboutModule.cpp
blob: 92bc975f48e5bd37f0068388fb8ea95244c98a40 (plain) (tree)
1
2
3
4
5



                                                   
                                  


















                                                                              

                   









                                                          
                     
                     
                       
                                  


                            
                                     

                             
                     
                      
                                  
 
                       
 
                                
 
                            
 
                                  
 
                                                  

 
                                   
 
                                                  

 
                                                    


                                             

                                                 


                            


                             
                                                                         
         
                                                       
         
 





                                                                        






                                                                                                      










                                                


                       

                                                 
 

                          
                    

                                                    


                                         
                                                               











                                                                                   

                                              
 


                                    
 

        





                                                     
 
                                                                                                            

















                                                    
                                                                      

                                      
                                                                 




                                                                               



                                                                               







                                                                                   
 



                     

                                                    

                                                   



                                                   
 

                                             




                                                     
                                                                                 

                                                                                                  
                                                                                                     
                                                                           
                                                                                                                    
                                                   





                                                      
                                                                                                  

                                                                                                  

                                                                                                





                                                     
                                                                             


                                                                                    




                                                                                                 


                                                           

                                     
 
                                                                             
                                                 

                                                              
                                                                             


                                                                       



                                                          












































                                                                                                                            


                                                      


                                     
                                                                             



                                                                             





                                                                                       





                                                    
                                                                             



                                                                             




                                                                      
                                                                                  

                                                                                       


                                                      
                                                          
                                                             



                                                                                    


                                     
                                                                             



                                                                             




                                                                                          


                                                           


                                     
                                                                             



                                                                                




                                                                                 


                                                  


                                     
                                                                             



                                                                             







                                                                                       


                                                    





                                                                                                    


                                                          





                                      
                                                                             



                                                                              







                                                                                       



                                                               
                                                                            
                                                                               
                                                                    


                                                                              
         
                                              

                                                              
                                                                    


                                                                                                








                                              

                                                      
 

                          
                                       
                    

                                       
                                               
 

                                                                        

                                                                                            

                             
                                                                  
                                                                    
                                   
 



                                                                            
 




                                                                           

                                         
                                           
 
                                                                                      

                                                                           
 
                                                  
         
                                   

                                                                

                                                                                                                                               

                                   
                       

                         


                                
 




                                                        

                          











                                                                        
                                                                 






                                                                                 

                                                                   
                                                                            
                                                                             














                                                                                          
                                               






                                                     
                                                
 

                          
                    
                                                 
                                                                                        


                                         
                                                                         

















                                                                                           
                               




                                                                                                             
                              





                                           



                                                                          


                                                              
 





                                                         







                                                                                        
                                   
                        
                                 
                      

                          
         







                                                  
         

                                                     


                                              
         

                                       
                                      
                                           
                                         
                                                                                      
                                          
                                      


                                                                                         


                                                                              
                                          
                                      


                                                                                           




                                            
                      




                            


                                             
                                                                             

                                   





                                                                                    
                                   

                                                    






                                                                          
                                                                                              




                                          
                                
 
                     


        

                                                



                                                                     
/*
 *  Copyright (C) 2001 Matt Aubury, Philip Langdale
 *  Copyright (C) 2004 Crispin Flowerday
 *  Copyright (C) 2005 Christian Persch
 *  Copyright (C) 2005 Adam Hooper
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 2, or (at your option)
 *  any later version.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with this program; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 *
 *  $Id$
 */

#include "mozilla-config.h"
#include "config.h"

#include <string.h>

#include <glib/gi18n.h>
#include <gtk/gtk.h>

// we need nsEscape which depends on internal strings :(((
#define MOZILLA_INTERNAL_API 1
#include <nsString.h>

#include <nsAutoPtr.h>
#include <nsCOMPtr.h>
#include <nsEscape.h>
#include <nsIChannel.h>
#include <nsIInputStreamChannel.h>
#include <nsIInputStream.h>
#include <nsIIOService.h>
#include <nsIOutputStream.h>
#include <nsIScriptSecurityManager.h>
#include <nsIStorageStream.h>
#include <nsIURI.h>
#include <nsNetCID.h>
#include <nsNetUtil.h>
#include <nsServiceManagerUtils.h>

#include "ephy-debug.h"

#include "EphyRedirectChannel.h"

#include "EphyAboutModule.h"

EphyAboutModule::EphyAboutModule()
{
    LOG ("EphyAboutModule ctor [%p]\n", this);
}

EphyAboutModule::~EphyAboutModule()
{
    LOG ("EphyAboutModule dtor [%p]\n", this);
}

NS_IMPL_ISUPPORTS1 (EphyAboutModule, nsIAboutModule)

/* nsIChannel newChannel (in nsIURI aURI); */
NS_IMETHODIMP
EphyAboutModule::NewChannel(nsIURI *aURI,
                nsIChannel **_retval)
{
    NS_ENSURE_ARG(aURI);

    nsCAutoString path;
    aURI->GetPath (path);

    if (strncmp (path.get(), "neterror?", strlen ("neterror?")) == 0)
    {
        return CreateErrorPage (aURI, _retval);
    }

    if (strncmp (path.get (), "recover?", strlen ("recover?")) == 0)
    {
        return CreateRecoverPage (aURI, _retval);
    }

    if (strcmp (path.get (), "epiphany") == 0)
    {
        return Redirect (nsDependentCString ("file://" SHARE_DIR "/epiphany.xhtml"), _retval);
    }

    return NS_ERROR_ILLEGAL_VALUE;
}

#ifdef HAVE_GECKO_1_9
/* unsigned long getURIFlags(in nsIURI aURI); */
NS_IMETHODIMP
EphyAboutModule::GetURIFlags (nsIURI *aURI,
                  PRUint32 *_result)
{
    *_result = 0;
    return NS_OK;
}
#endif

/* private functions */

nsresult
EphyAboutModule::Redirect(const nsACString &aURL,
              nsIChannel **_retval)
{
    *_retval = nsnull;

    nsresult rv;
    nsCOMPtr<nsIURI> uri;
    rv = NS_NewURI (getter_AddRefs (uri), aURL);
    NS_ENSURE_SUCCESS (rv, rv);

    nsCOMPtr<nsIChannel> tempChannel;
    rv = NS_NewChannel (getter_AddRefs (tempChannel), uri);
    NS_ENSURE_SUCCESS (rv, rv);

    tempChannel->SetOriginalURI (uri);

    nsCOMPtr<nsIScriptSecurityManager> securityManager = 
            do_GetService(NS_SCRIPTSECURITYMANAGER_CONTRACTID, &rv);
    NS_ENSURE_SUCCESS (rv, rv);

    nsCOMPtr<nsIPrincipal> principal;
    rv = securityManager->GetCodebasePrincipal(uri, getter_AddRefs(principal));
    NS_ENSURE_SUCCESS (rv, rv);

    rv = tempChannel->SetOwner(principal);
    NS_ENSURE_SUCCESS (rv, rv);

    tempChannel.swap (*_retval);

    return NS_OK;
}

nsresult
EphyAboutModule::ParseURL(const char *aURL,
              nsACString &aCode,
              nsACString &aRawOriginURL,
              nsACString &aOriginURL,
              nsACString &aOriginCharset,
              nsACString &aTitle)
{
    /* The page URL is of the form "about:neterror?e=<errorcode>&u=<URL>&c=<charset>&d=<description>" */
    const char *query = strstr (aURL, "?");
    if (!query) return NS_ERROR_FAILURE;
    
    /* skip the '?' */
    ++query;
    
    char **params = g_strsplit (query, "&", -1);
    if (!params) return NS_ERROR_FAILURE;
    
    for (PRUint32 i = 0; params[i] != NULL; ++i)
    {
        char *param = params[i];
    
        if (strlen (param) <= 2) continue;
    
        switch (param[0])
        {
            case 'e':
                aCode.Assign (nsUnescape (param + 2));
                break;
            case 'u':
                aRawOriginURL.Assign (param + 2);
                aOriginURL.Assign (nsUnescape (param + 2));
                break;
            case 'c':
                aOriginCharset.Assign (nsUnescape (param + 2));
                break;
            /* The next one is not used in neterror but recover: */
            case 't':
                aTitle.Assign (nsUnescape (param + 2));
                break;
            case 'd':
                /* we don't need mozilla's description parameter */
            default:
                break;
        }
    }

    g_strfreev (params);

    return NS_OK;
}

nsresult
EphyAboutModule::GetErrorMessage(nsIURI *aURI,
                 const char *aError,
                 char **aStockIcon,
                 char **aTitle,
                 char **aPrimary,
                 char **aSecondary,
                 char **aTertiary,
                 char **aLinkIntro)
{
    *aStockIcon = GTK_STOCK_DIALOG_ERROR;

    if (strcmp (aError, "protocolNotFound") == 0)
    {
        nsCAutoString scheme;
        aURI->GetScheme (scheme);

        /* Translators: %s is the name of a protocol, like "http" etc. */
        *aTitle = g_strdup_printf (_("“%s” Protocol is not Supported"), scheme.get());
        /* Translators: %s is the name of a protocol, like "http" etc. */
        *aPrimary = g_strdup_printf (_("“%s” protocol is not supported."), scheme.get());
        /* FIXME: get the list of supported protocols from necko */
        *aSecondary = _("Supported protocols are “http”, “https”, “ftp”, “file”, “smb” "
                "and “sftp”.");
    }
    else if (strcmp (aError, "fileNotFound") == 0)
    {
        nsCAutoString path;
        aURI->GetPath (path);

        /* Translators: %s is the path and filename, for example "/home/user/test.html" */
        *aTitle = g_markup_printf_escaped (_("File “%s” not Found"), path.get());
        /* Translators: %s is the path and filename, for example "/home/user/test.html" */
        *aPrimary = g_markup_printf_escaped (_("File “%s” not found."), path.get());
        *aSecondary = _("Check the location of the file and try again.");
    }
    else if (strcmp (aError, "dnsNotFound") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped (_("“%s” Could not be Found"),
                             host.get());
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped (_("“%s” could not be found."),
                             host.get());
        *aSecondary = _("Check that you are connected to the internet, and "
                "that the address is correct.");
        *aLinkIntro = _("If this page used to exist, you may find an archived version:");
    }
    else if (strcmp (aError, "connectionFailure") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped
            (_("“%s” Refused the Connection"),
             host.get());
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped
                (_("“%s” refused the connection."),
                 host.get());

        /* FIXME what about 127.0.0.* ? */
        if (strcmp (host.get(), "localhost") == 0)
        {
            PRInt32 port;
            aURI->GetPort (&port);

            *aSecondary = _("Likely causes of the problem are");

            /* Try to get the service name attached to that port */
            if (port != -1)
            {
                struct servent *serv;
            
                if ((serv = getservbyport (htons (port), NULL)) != NULL)
                {
                    *aTertiary = g_markup_printf_escaped (
                            _("<ul>"
                               "<li>the service ""%s"" isn't started.</li>"
                               "Try to start it using the Services Configuration Tool in "
                               "Desktop > System Settings > Server Settings > Services, or</ul>"
                               "<ul><li>the port number %d is wrong.</li>"
                               "</ul>"),
                            serv->s_name, port);
                }
                else
                {
                    *aTertiary = g_markup_printf_escaped (
                            _("<ul>"
                               "<li>some service isn't started, or</li>"
                               "<li>the port number %d is wrong.</li>"
                               "</ul>"),
                            port);
                }
            }
            else
            {
                *aTertiary = _("<ul>"
                           "<li>some service isn't started, or</li>"
                           "<li>you got the port number wrong.</li>"
                           "</ul>");
            }
        }
        else
        {
            *aSecondary = _("The server may be busy or you may have a "
                    "network connection problem. Try again later.");
            *aLinkIntro = _("There may be an old version of the page you wanted:");
        }
    }
    else if (strcmp (aError, "netInterrupt") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped
                (_("“%s” Interrupted the Connection"),
                 host.get());
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped
                (_("“%s” interrupted the connection."),
                 host.get());
        *aSecondary = _("The server may be busy or you may have a "
                "network connection problem. Try again later.");
        *aLinkIntro = _("There may be an old version of the page you wanted:");
    }
    else if (strcmp (aError, "netTimeout") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped
                (_("“%s” is not Responding"),
                 host.get());
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped
                (_("“%s” is not responding."),
                 host.get());
        *aSecondary = _("The connection was lost because the "
                "server took too long to respond.");
        *aTertiary = _("The server may be busy or you may have a network "
                    "connection problem. Try again later.");
        *aLinkIntro = _("There may be an old version of the page you wanted:");
    }
    else if (strcmp (aError, "malformedURI") == 0)
    {
        *aTitle = g_strdup (_("Invalid Address"));
        *aPrimary = g_strdup (_("Invalid address."));
        *aSecondary = g_strdup (_("The address you entered is not valid."));
    }
    else if (strcmp (aError, "redirectLoop") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped
                (_("“%s” Redirected Too Many Times"),
                 host.get());
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped
                (_("“%s” redirected too many times."),
                 host.get());
        *aSecondary = _("The redirection has been stopped for security reasons.");
        *aLinkIntro = _("There may be an old version of the page you wanted:");
    }
    else if (strcmp (aError, "unknownSocketType") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped
                (_("“%s” Requires an Encrypted Connection"),
                 host.get());
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped
                (_("“%s” requires an encrypted connection."),
                 host.get());
        *aSecondary = _("The document could not be loaded because "
                "encryption support is not installed.");
    }
    else if (strcmp (aError, "netReset") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped
                (_("“%s” Dropped the Connection"),
                 host.get());
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped
                (_("“%s” dropped the connection."),
                 host.get());
        *aSecondary = _("The server dropped the connection "
                 "before any data could be read.");
        *aTertiary = _("The server may be busy or you may have a "
                   "network connection problem. Try again later.");
        *aLinkIntro = _("There may be an old version of the page you wanted:");
    }
    else if (strcmp (aError, "netOffline") == 0)
    {
        /* Error is a bit too strong here */
        *aStockIcon = GTK_STOCK_DIALOG_INFO;

        *aTitle = g_strdup (_("Cannot Load Document Whilst Working Offline"));
        *aPrimary = g_strdup (_("Cannot load document whilst working offline."));
        *aSecondary = _("To view this document, disable “Work Offline” and try again.");
    }
    else if (strcmp (aError, "deniedPortAccess") == 0)
    {
        nsCAutoString host;
        aURI->GetHost (host);

        PRInt32 port = -1;
        aURI->GetPort (&port);

        /* Translators: %s is the hostname, like "www.example.com" */
        *aTitle = g_markup_printf_escaped
                (_("“%s” Denied Access to Port “%d”"),
                 host.get(), port > 0 ? port : 80);
        /* Translators: %s is the hostname, like "www.example.com" */
        *aPrimary = g_markup_printf_escaped
                (_("“%s” denied access to port “%d”."),
                 host.get(), port > 0 ? port : 80);
        *aSecondary = g_strdup (_("The server dropped the connection "
                      "before any data could be read."));
        *aTertiary = _("The server may be busy or you may have a "
                   "network connection problem. Try again later.");
        *aLinkIntro = _("There may be an old version of the page you wanted:");
    }
    else if (strcmp (aError, "proxyResolveFailure") == 0 ||
         strcmp (aError, "proxyConnectFailure") == 0)
    {
        *aTitle = g_strdup (_("Could not Connect to Proxy Server"));
        *aPrimary = g_strdup (_("Could not connect to proxy server."));
        *aSecondary = _("Check your proxy server settings. "
                "If the connection still fails, there may be "
                "a problem with your proxy server or your "
                "network connection.");
    }
    /* This was introduced in gecko 1.9 */
    else if (strcmp (aError, "contentEncodingError") == 0)
    {
        *aTitle = g_strdup (_("Could not Display Content"));
        *aPrimary = g_strdup (_("Could not display content."));
        *aSecondary = _("The page uses an unsupported or invalid form of compression.");
    }
    else
    {
        return NS_ERROR_ILLEGAL_VALUE;
    }

    return NS_OK;
}

nsresult
EphyAboutModule::CreateErrorPage(nsIURI *aErrorURI,
                 nsIChannel **_retval)
{
    *_retval = nsnull;

        /* First parse the arguments */
    nsresult rv;
        nsCAutoString spec;
    rv = aErrorURI->GetSpec (spec);
    NS_ENSURE_TRUE (NS_SUCCEEDED (rv), rv);

    nsCAutoString error, rawurl, url, charset, dummy;
    rv = ParseURL (spec.get (), error, rawurl, url, charset, dummy);
    if (NS_FAILED (rv)) return rv;
    if (error.IsEmpty () || rawurl.IsEmpty () || url.IsEmpty()) return NS_ERROR_FAILURE;

    nsCOMPtr<nsIURI> uri;
    rv = NS_NewURI (getter_AddRefs (uri), url, charset.get());
    /* FIXME can uri be NULL if the original url was invalid? */
    NS_ENSURE_SUCCESS (rv, rv);

    char *stock_id = nsnull, *title = nsnull, *primary = nsnull,
         *secondary = nsnull, *tertiary = nsnull, *linkintro = nsnull;
    rv = GetErrorMessage (uri, error.get(), &stock_id, &title, &primary,
                  &secondary, &tertiary, &linkintro);

    /* we don't know about this error code.
     * FIXME: We'd like to forward to mozilla's about:neterror handler,
     * but I don't know how to. So just redirect to the same page that
     * mozilla's handler redirects to.
     */
    if (rv == NS_ERROR_ILLEGAL_VALUE)
    {
        nsCAutoString newurl(spec);

        /* remove "about:neterror" part and insert mozilla's error page url */
        newurl.Cut(0, strlen ("about:neterror"));
        newurl.Insert("chrome://global/content/netError.xhtml", 0);

        return Redirect (newurl, _retval);
    }
    NS_ENSURE_SUCCESS (rv, rv);
    NS_ENSURE_TRUE (primary && secondary, NS_ERROR_FAILURE);

    nsCOMPtr<nsIChannel> channel;
    rv = WritePage (aErrorURI, uri, aErrorURI, rawurl, title, stock_id, primary, secondary, tertiary, linkintro, getter_AddRefs (channel));
    NS_ENSURE_SUCCESS (rv, rv);

    g_free (title);
    g_free (primary);

    channel.swap (*_retval);

    return NS_OK;
}

nsresult
EphyAboutModule::CreateRecoverPage(nsIURI *aRecoverURI,
                   nsIChannel **_retval)
{
    *_retval = nsnull;

        /* First parse the arguments */
    nsresult rv;
        nsCAutoString spec;
    rv = aRecoverURI->GetSpec (spec);
    NS_ENSURE_TRUE (NS_SUCCEEDED (rv), rv);

    nsCAutoString error, rawurl, url, charset, title;
    rv = ParseURL (spec.get (), error, rawurl, url, charset, title);
    if (NS_FAILED (rv)) return rv;
    if (rawurl.IsEmpty () || url.IsEmpty()) return NS_ERROR_FAILURE;

    nsCOMPtr<nsIURI> uri;
    rv = NS_NewURI(getter_AddRefs (uri), url, charset.get());
    NS_ENSURE_SUCCESS (rv, rv);

    char *secondary = g_strdup_printf
        (_("The page “%s” in this tab was not fully loaded yet when "
           "the web browser crashed; it could have caused the crash."),
         url.get());

    nsCOMPtr<nsIChannel> channel;
    rv = WritePage (aRecoverURI, uri, uri, rawurl, title.get(),
            GTK_STOCK_DIALOG_INFO, title.get() /* as primary */,
            secondary, nsnull, nsnull, getter_AddRefs (channel));
    NS_ENSURE_SUCCESS (rv, rv);

    nsRefPtr<EphyRedirectChannel> redirectChannel (new EphyRedirectChannel (channel));
    if (!redirectChannel) return NS_ERROR_OUT_OF_MEMORY;

    g_free (secondary);

    NS_ADDREF(*_retval = redirectChannel);

    return NS_OK;
}

nsresult
EphyAboutModule::WritePage(nsIURI *aOriginalURI,
               nsIURI *aURI,
               nsIURI *aChannelURI,
               const nsACString &aRawURL,
               const char *aTitle,
               const char *aStockIcon,
               const char *aPrimary,
               const char *aSecondary,
               const char *aTertiary,
               const char *aLinkIntro,
               nsIChannel **_retval)
{
    *_retval = nsnull;

    nsresult rv;
    nsCOMPtr<nsIStorageStream> storageStream;
    rv = NS_NewStorageStream (16384, (PRUint32) -1, getter_AddRefs (storageStream));
    NS_ENSURE_SUCCESS (rv, rv);

    nsCOMPtr<nsIOutputStream> stream;
    rv = storageStream->GetOutputStream (0, getter_AddRefs (stream));
    NS_ENSURE_SUCCESS (rv, rv);

    char *language = g_strdup (pango_language_to_string (gtk_get_default_language ()));
    g_strdelimit (language, "_-@", '\0');

    Write (stream,
           "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n"
           "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\" "
           "\"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">\n"
           "<html xmlns=\"http://www.w3.org/1999/xhtml\" lang=\"");
    Write (stream, language);
    Write (stream,
           "\" xml:lang=\"");
    Write (stream, language);
    Write (stream,
           "\">\n"
           "<head>\n"
        "<title>");
    Write (stream, aTitle);
    /* no favicon for now, it would pollute the favicon cache */
    /* "<link rel=\"icon\" type=\"image/png\" href=\"moz-icon://stock/gtk-dialog-error?size=16\" />\n" */
    Write (stream,
        "</title>\n"
        "<style type=\"text/css\">\n"
        "div#body {\n"
            "top: 12px;\n"
            "right: 12px;\n"
            "bottom: 12px;\n"
            "left: 12px;\n"
            "overflow: auto;\n"

            "background: -moz-dialog url('moz-icon://stock/");
    Write (stream, aStockIcon);
    Write (stream,
            "?size=dialog') no-repeat 12px 12px;\n"
            "color: -moz-dialogtext;\n"
            "font: message-box;\n"
            "border: 1px solid -moz-dialogtext;\n"

            "padding: 12px 12px 12px 72px;\n"
        "}\n"

        "h1 {\n"
            "margin: 0;\n"
            "font-size: 1.2em;\n"
        "}\n"
        "</style>\n"
           "</head>\n"
           "<body dir=\"");
    Write (stream,
           gtk_widget_get_default_direction () == GTK_TEXT_DIR_RTL ? "rtl" : "ltr");
    Write (stream,
           "\">\n"
        "<div id=\"body\">"
        "<h1>");
    Write (stream, aPrimary);
    Write (stream,
           "</h1>\n");
    if (aSecondary)
    {
        Write (stream, "<p>");
        Write (stream, aSecondary);
        if (aTertiary)
        {
            Write (stream, " ");
            Write (stream, aTertiary);
        }
        Write (stream, "</p>\n");
    }

    PRBool isHttp = PR_FALSE, isHttps = PR_FALSE;
    aURI->SchemeIs ("http", &isHttp);
    aURI->SchemeIs ("https", &isHttps);
    if (aLinkIntro && (isHttp || isHttps))
    {
        nsCString raw(aRawURL);

        Write (stream, "<p>");
        Write (stream, aLinkIntro);
        Write (stream, "<ul>\n");
        Write (stream, "<li><a href=\"http://www.google.com/search?q=cache:");
        Write (stream, raw.get());
        Write (stream, "\">");
        /* Translators: The text before the "|" is context to help you decide on
         * the correct translation. You MUST OMIT it in the translated string. */
        Write (stream, Q_("You may find an old version:|in the Google Cache"));
        Write (stream, "</a></li>\n");

        Write (stream, "<li><a href=\"http://web.archive.org/web/*/");
        Write (stream, raw.get());
        Write (stream, "\">");
        /* Translators: The text before the "|" is context to help you decide on
         * the correct translation. You MUST OMIT it in the translated string. */
        Write (stream, Q_("You may find an old version:|in the Internet Archive"));
        Write (stream, "</a></li>\n"
                   "</ul>\n"
                   "</p>");
    }

    Write (stream,
        "</div>\n"
           "</body>\n"
           "</html>\n");

    g_free (language);
 
    /* finish the rendering */
    nsCOMPtr<nsIInputStream> inputStream;
    rv = storageStream->NewInputStream (0, getter_AddRefs (inputStream));
    NS_ENSURE_SUCCESS (rv, rv);

    nsCOMPtr<nsIChannel> channel;
    rv = NS_NewInputStreamChannel (getter_AddRefs (channel),
                       aChannelURI,
                       inputStream,
                       NS_LITERAL_CSTRING ("application/xhtml+xml"),
                       NS_LITERAL_CSTRING ("utf-8"));
    NS_ENSURE_SUCCESS (rv, rv);
    
    rv = channel->SetOriginalURI (aOriginalURI);
    NS_ENSURE_SUCCESS (rv, rv);

    nsCOMPtr<nsIScriptSecurityManager> securityManager 
        (do_GetService(NS_SCRIPTSECURITYMANAGER_CONTRACTID, &rv));
    NS_ENSURE_SUCCESS (rv, rv);

    nsCOMPtr<nsIPrincipal> principal;
    rv = securityManager->GetCodebasePrincipal (aOriginalURI, getter_AddRefs (principal));
    NS_ENSURE_SUCCESS (rv, rv);

    rv = channel->SetOwner(principal);
    NS_ENSURE_SUCCESS (rv, rv);

    channel.swap (*_retval);

    return NS_OK;
}

nsresult
EphyAboutModule::Write(nsIOutputStream *aStream,
               const char *aText)
{
    PRUint32 bytesWritten;
    return aStream->Write (aText, strlen (aText), &bytesWritten);
}