From a8fe5e52e109c55b56927ec8beea795c9521d1bb Mon Sep 17 00:00:00 2001
From: g0dil <g0dil@wiback.org>
Date: Thu, 15 Feb 2007 13:42:40 +0000
Subject: [PATCH] Add TCPv6 SocketHandle implementation

---
 Example.dox                      |  7 ++-
 Mainpage.dox                     | 58 ++++++++++---------
 Socket/INetAddressing.cc         |  8 ---
 Socket/INetAddressing.hh         | 98 ++++++++++++++++++++++----------
 Socket/INetProtocol.cc           | 18 +++++-
 Socket/INetProtocol.hh           | 14 ++++-
 Socket/TCPSocketHandle.cc        | 49 ++++++++++++++++
 Socket/TCPSocketHandle.hh        | 43 +++++++++++++-
 doclib/doxy-header-overview.html | 13 ++++-
 doclib/doxy-header.html          |  4 +-
 doclib/filter.pl                 |  3 +
 doclib/senf.css                  | 28 ++++-----
 12 files changed, 251 insertions(+), 92 deletions(-)

diff --git a/Example.dox b/Example.dox
index e42aac507..f2d86f65c 100644
--- a/Example.dox
+++ b/Example.dox
@@ -89,9 +89,10 @@
     application, check out the library, go to the \c Sniffer directory
     and execute
 
-    <pre class="fragment">
-      # scons -u
-      # ./Sniffer</pre>
+    <pre>
+    # scons -u
+    # ./Sniffer
+    </pre>
 
     \see \ref components \n
          \ref build \n
diff --git a/Mainpage.dox b/Mainpage.dox
index b47235462..03e5784f5 100644
--- a/Mainpage.dox
+++ b/Mainpage.dox
@@ -29,10 +29,7 @@
     mailing lists</a>.
 
     \see \ref usage\n
-         \ref example\n
-         <a href="xref.html">Current status: Cross reference of action points</a>\n
-         <a class="ext" href="http://developer.berlios.de/projects/senf">The BerliOS project page</a>\n
-         <a class="ext" href="http://openfacts.berlios.de/index-en.phtml?title=SENF+Network+Framework">The SENF Wiki at BerliOS</a>
+         \ref example
 */
 
 /** \page usage Using the SENF framework
@@ -94,8 +91,9 @@
     repository. Change to your development directory and use the
     following subversion command
 
-    <pre class="fragment">
-      $ svn checkout http://svn.berlios.de/svnroot/repos/senf/trunk senf</pre>
+    <pre>
+    $ svn checkout http://svn.berlios.de/svnroot/repos/senf/trunk senf
+    </pre>
 
     This will create a new directory \c senf within the current
     directory. For further documentation on the use of Subversion, see
@@ -109,9 +107,10 @@
     To build the library, execute all unit tests and build the Sniffer
     test application, use
 
-    <pre class="fragment">
-      $ scons
-      $ scons all_tests</pre>
+    <pre>
+    $ scons
+    $ scons all_tests
+    </pre>
 
     in the \c senf directory. This assumes, that you want to build the
     library with your default gcc and requires the boost libraries to
@@ -223,8 +222,9 @@
     check out the needed directories from the BerliOS SENF
     repository. Change to the 'Foo' directory and type
     
-      <pre class="fragment">
-        $ svn propedit svn:externals .</pre>
+    <pre>
+    $ svn propedit svn:externals .
+    </pre>
 
     The default editor (probably VI) will be started with the current
     value of the svn:externals property (which will probably be
@@ -236,16 +236,18 @@
     For example, if you want to use the \c Scheduler and \c Socket
     module, the file will look like
 
-      <pre class="fragment">
-        senfscons http://svn.berlios.de/svnroot/repos/senf/trunk/senfscons
-        Utils http://svn.berlios.de/svnroot/repos/senf/trunk/Utils
-        Scheduler http://svn.berlios.de/svnroot/repos/senf/trunk/Scheduler
-        Socket http://svn.berlios.de/svnroot/repos/senf/trunk/Socket</pre>
+    <pre>
+    senfscons http://svn.berlios.de/svnroot/repos/senf/trunk/senfscons
+    Utils http://svn.berlios.de/svnroot/repos/senf/trunk/Utils
+    Scheduler http://svn.berlios.de/svnroot/repos/senf/trunk/Scheduler
+    Socket http://svn.berlios.de/svnroot/repos/senf/trunk/Socket
+    </pre>
 
     exit the editor and the property will be set. Now run
 
-      <pre class="fragment">
-        $ svn update</pre>
+    <pre>
+    $ svn update
+    </pre>
 
     and the code will be checked out into the corresponding
     directories. 
@@ -268,8 +270,9 @@
     add \c SConfig to the list of files ignored by Subversion in the
     project root. In the project root execute
 
-    <pre class="fragment">
-      $ svn propedit svn:ignore .</pre>
+    <pre>
+    $ svn propedit svn:ignore .
+    </pre>
 
     and add \c SConfig as a new line to the property.
 
@@ -277,21 +280,24 @@
 
     You should now be able to build your project using
 
-    <pre class="fragment">
-      $ scons</pre>
+    <pre>
+    $ scons
+    </pre>
 
     If you have not changed the \c SConstruct file, this will build
     all modules you have importet into your project. To build and
     execute the unit tests, use
 
-    <pre class="fragment">
-      $ scons all_tests</pre>
+    <pre>
+    $ scons all_tests
+    </pre>
 
     you can also build only a subdirectory by changing to it and
     running
     
-    <pre class="fragment">
-      $ scons -u [target]</pre>
+    <pre>
+    $ scons -u [target]
+    </pre>
 
     \see <a href="../../senfscons/doc/html/index.html">SENFSCons reference</a> \n
          <a class="ext" href="http://www.scons.org/documentation.php">SCons documentation</a> \n
diff --git a/Socket/INetAddressing.cc b/Socket/INetAddressing.cc
index b1efabc43..301d86cdd 100644
--- a/Socket/INetAddressing.cc
+++ b/Socket/INetAddressing.cc
@@ -148,14 +148,6 @@ prefix_ void senf::INet6SocketAddress::clear()
     sockaddr_.sin6_family = AF_INET6;
 }
 
-prefix_ void senf::INet6SocketAddress::address(std::string const & addr)
-{
-    if (addr[0]=='[')
-	assignAddr(addr);
-    else
-	host(addr);
-}
-
 prefix_ std::string senf::INet6SocketAddress::address()
     const
 {
diff --git a/Socket/INetAddressing.hh b/Socket/INetAddressing.hh
index d97c83cd1..4a8deca62 100644
--- a/Socket/INetAddressing.hh
+++ b/Socket/INetAddressing.hh
@@ -107,7 +107,16 @@ namespace senf {
 
     /** \brief IPv6 network address
 
-	\todo Implement
+	INet6Address represents a 128bit IPv6 network address. This class supports all standard
+	numeric string representations of IPv6 addresses. This class does not integrate with \c
+	gethostbyname() and so does not support host names.
+
+	The conversion constructors allow the use of string constants whereever an INet6Address is
+	expected. Especially, it is possible to assign a string to an address to change it's value.
+
+	\implementation The <tt>char const *</tt> constructor overload is needed to support
+	    string-literals where an INet6Address is expected (the C++ standard does not allow
+	    chaining conversion constructors like char const * -> std::string -» INet6Address)
      */
     class INet6Address
     {
@@ -127,17 +136,19 @@ namespace senf {
         ///@}
         ///////////////////////////////////////////////////////////////////////////
 
-	void clear();
-	std::string address() const;
+	void clear();                   ///< Clear address
+	std::string address() const;    ///< Return printable address representation
 
-	bool operator==(INet6Address const & other) const;
-	bool operator!=(INet6Address const & other) const;
+	bool operator==(INet6Address const & other) const; ///< Compare addresses for equality
+	bool operator!=(INet6Address const & other) const; ///< Inverse of above
 
-	struct in6_addr & addr();
-	struct in6_addr const & addr() const;
-	struct in6_addr * addr_p();
+	struct in6_addr & addr();       ///< Access internal address representation
+	struct in6_addr const & addr() const; 
+                                        ///< Access internal address representation in const context
+	struct in6_addr * addr_p();     ///< Get pointer to internal address repr
 	struct in6_addr const * addr_p() const;
-	unsigned addr_len() const;
+                                        ///< Get const pointer to internal address repr
+	unsigned addr_len() const;      ///< Size of an IPv6 address (16 bytes)
 
     protected:
 
@@ -145,14 +156,40 @@ namespace senf {
 	struct in6_addr addr_;
     };
 
+    
+    /** \brief Output INet6Address instance as it's string representation
+     */
     std::ostream & operator<<(std::ostream & os, INet6Address const & addr);
 
     /** \brief IPv6 socket address
 
+	This class wrapps the standard \c sockaddr_in6 structure. INet6SocketAddress provides access
+	to all members of the sockaddr_in6 structure. Additionally, INet6SocketAddress supports the
+	string representation
+
+	\par "" <tt>[</tt> <i>address</i> [ <tt>\@</tt> <i>interface</i> ] <tt>]:</tt> <i>port</i>
+
+	Where \e address is an arbitrary numeric IPv6 address, \e interface is an optional network
+	interface name and \e port is the port number. The interface specification is only valid if
+	\e address is link-local address. The URL representation of an IPv6 address is as above
+	without the optional interface spec.
+
+	INet6SocketAddress supports conversion constructors from it's string
+	representation. Therefore, wherever a INet6SocketAddress instance is expected, a string may
+	be used instead.
+
 	\implementation The sockaddr_in6 structure has an sin6_flowinfo member. However RFC3493 does
-	not give the use of this field and specifies, that the field should be ignored ... so that's
-	what we do. Furthermore, the GNU libc reference states, that this field is not implemented
-	in the library.
+	    not give the use of this field and specifies, that the field should be ignored ... so
+	    that's what we do. Furthermore, the GNU libc reference states, that this field is not
+	    implemented in the library.
+
+	\implementation We need to return the address in host() by value since we need to return a
+	    INet6Address. However, sockaddr_in6 does not have one ...
+
+	\implementation The <tt>char const *</tt> constructor overload is needed to support
+	    string-literals where an INet6SocketAddress is expected (the C++ standard does not allow
+	    chaining conversion constructors like <tt>char const *</tt> -> \c std::string -> \c
+	    INet6SocketAddress) 
 
 	\idea Implement a INet6Address_ref class which has an interface identical to INet6Address
 	and is convertible to INet6Address (the latter has a conversion constructor taking the
@@ -170,32 +207,35 @@ namespace senf {
         ///\name Structors and default members
         ///@{
 
-        INet6SocketAddress();
-        INet6SocketAddress(std::string const & addr);
-        INet6SocketAddress(char const * addr);
+        INet6SocketAddress();           ///< Create empty instance
+        INet6SocketAddress(std::string const & addr); 
+                                        ///< Initialize/convert from string represenation
+        INet6SocketAddress(char const * addr); ///< Same as above to support string literals
 	INet6SocketAddress(INet6Address const & addr, unsigned port);
+                                        ///< Initialize from address and port
 	INet6SocketAddress(INet6Address const & addr, unsigned port, std::string const & iface);
+                                        ///< Initialize explicitly from given parameters
 	INet6SocketAddress(std::string const & addr, std::string const & iface);
+                                        ///< Initialize from URL representation and explit interface
 
         ///@}
         ///////////////////////////////////////////////////////////////////////////
 
-	bool operator==(INet6SocketAddress const & other) const;
-	bool operator!=(INet6SocketAddress const & other) const;
+	bool operator==(INet6SocketAddress const & other) const; ///< Check addresses for equality
+	bool operator!=(INet6SocketAddress const & other) const; ///< Inverse of above
 
-	void clear();
+	void clear();                   ///< Clear socket address
 
-	std::string address() const;
-	void address(std::string const & addr);
+	std::string address() const;    ///< Get printable address representation
 
-	INet6Address host() const;
-	void host(INet6Address const & addr);
+	INet6Address host() const;      ///< Get address
+	void host(INet6Address const & addr); ///< Change address
 	
-	unsigned port() const;
-	void port(unsigned poirt);
+	unsigned port() const;          ///< Get port number
+	void port(unsigned poirt);      ///< Change port number
 	
-	std::string iface() const;
-	void iface(std::string const & iface);
+	std::string iface() const;      ///< Get interface name
+	void iface(std::string const & iface); ///< Change interface
 	
         ///\name Generic SocketAddress interface
         ///@{
@@ -215,6 +255,8 @@ namespace senf {
 	struct sockaddr_in6 sockaddr_;
     };
 
+    /** \brief Output INet6SocketAddress instance as it's string representation
+     */
     std::ostream & operator<<(std::ostream & os, INet6SocketAddress const & addr);
 
     /** \brief Signal invalid INet address syntax
@@ -257,7 +299,7 @@ namespace senf {
     /** \brief Addressing policy supporting IPv6 addressing
 	
 	\par Address Type:
-	    INet6Address
+	    INet6SocketAddress
 	
 	This addressing policy implements addressing using Internet V6
 	addresses.
@@ -265,8 +307,6 @@ namespace senf {
 	The various members are directly importet from
 	GenericAddressingPolicy which see for a detailed
 	documentation.
-
-	\todo implement
      */
     struct INet6AddressingPolicy
         : public AddressingPolicyBase,
diff --git a/Socket/INetProtocol.cc b/Socket/INetProtocol.cc
index 40063693d..ca93fa954 100644
--- a/Socket/INetProtocol.cc
+++ b/Socket/INetProtocol.cc
@@ -37,7 +37,7 @@
 ///////////////////////////////cc.p////////////////////////////////////////
 
 ///////////////////////////////////////////////////////////////////////////
-// senf::INet4Protocol
+// senf::IPv4Protocol
 
 prefix_ void senf::IPv4Protocol::connect(INet4Address const & address)
     const
@@ -148,6 +148,22 @@ prefix_ void senf::IPv4Protocol::mcTTL(unsigned value)
         throw SystemException(errno);
 }
 
+///////////////////////////////////////////////////////////////////////////
+// senf::IPv6Protocol
+
+prefix_ void senf::IPv6Protocol::connect(INet6SocketAddress const & address)
+    const
+{
+    if (::connect(body().fd(),address.sockaddr_p(), address.sockaddr_len()) < 0)
+        throw SystemException(errno);
+}
+
+prefix_ void senf::IPv6Protocol::bind(INet6SocketAddress const & address)
+    const
+{
+    if (::bind(body().fd(),address.sockaddr_p(), address.sockaddr_len()) < 0)
+        throw SystemException(errno);
+}
 
 ///////////////////////////////cc.e////////////////////////////////////////
 #undef prefix_
diff --git a/Socket/INetProtocol.hh b/Socket/INetProtocol.hh
index 482f88ed7..e1116bd3f 100644
--- a/Socket/INetProtocol.hh
+++ b/Socket/INetProtocol.hh
@@ -127,12 +127,20 @@ namespace senf {
 	
 	This protocol facet introduces all the socket api protocol members which are related to IPv6
 	addressing.
-
-	\todo implement
      */
     class IPv6Protocol
         : public virtual SocketProtocol
-    {};
+    {
+    public:
+        void connect(INet6SocketAddress const & address) const; ///< Connect to remote address
+                                        /**< \todo make this obsolete by allowing access to the
+					     ClientSocketHandle from ConcreateSocketProtocol 
+					     \param[in] address Address to connect to */
+        void bind(INet6SocketAddress const & address) const; ///< Set local socket address
+                                        /**< \todo make this obsolete by allowing access to the
+					     ClientSocketHandle from ConcreateSocketProtocol 
+					     \param[in] address Address to set */
+    };
 
     /// @}
 
diff --git a/Socket/TCPSocketHandle.cc b/Socket/TCPSocketHandle.cc
index 7202fe384..3023e2a34 100644
--- a/Socket/TCPSocketHandle.cc
+++ b/Socket/TCPSocketHandle.cc
@@ -38,6 +38,9 @@
 #define prefix_
 ///////////////////////////////cc.p////////////////////////////////////////
 
+///////////////////////////////////////////////////////////////////////////
+// senf::TCPv4SocketProtocol
+
 prefix_ void senf::TCPv4SocketProtocol::init_client()
     const
 {
@@ -81,6 +84,52 @@ prefix_ std::auto_ptr<senf::SocketProtocol> senf::TCPv4SocketProtocol::clone()
     return std::auto_ptr<SocketProtocol>(new TCPv4SocketProtocol());
 }
 
+///////////////////////////////////////////////////////////////////////////
+// senf::TCPv6SocketProtocol::
+
+prefix_ void senf::TCPv6SocketProtocol::init_client()
+    const
+{
+    int sock = ::socket(PF_INET6,SOCK_STREAM,0);
+    if (sock < 0)
+        throw SystemException(errno);
+    body().fd(sock);
+}
+
+prefix_ void
+senf::TCPv6SocketProtocol::init_client(INet6Address const & address)
+    const
+{
+    init_client();
+    connect(address);
+}
+
+prefix_ void senf::TCPv6SocketProtocol::init_server()
+    const
+{
+    int sock = ::socket(PF_INET,SOCK_STREAM,0);
+    if (sock < 0)
+        throw SystemException(errno);
+    body().fd(sock);
+}
+
+prefix_ void senf::TCPv6SocketProtocol::init_server(INet6Address const & address,
+                                                           unsigned backlog)
+    const
+{
+    init_server();
+    bind(address);
+    reuseaddr(true);
+    if (::listen(body().fd(),backlog) < 0)
+        throw SystemException(errno);
+}
+
+prefix_ std::auto_ptr<senf::SocketProtocol> senf::TCPv6SocketProtocol::clone()
+    const
+{
+    return std::auto_ptr<SocketProtocol>(new TCPv6SocketProtocol());
+}
+
 ///////////////////////////////cc.e////////////////////////////////////////
 #undef prefix_
 //#include "TCPSocketHandle.mpp"
diff --git a/Socket/TCPSocketHandle.hh b/Socket/TCPSocketHandle.hh
index 484e6222f..f94e43992 100644
--- a/Socket/TCPSocketHandle.hh
+++ b/Socket/TCPSocketHandle.hh
@@ -160,8 +160,6 @@ namespace senf {
 	ProtocolServerSocketHandle via the Socket Handle typedefs above.
 
 	\see TCPv4SocketProtocol
-	
-	\todo Implement
      */
     class TCPv6SocketProtocol
         : public ConcreteSocketProtocol<TCPv6Socket_Policy>, 
@@ -170,6 +168,47 @@ namespace senf {
           public BSDSocketProtocol,
           public AddressableBSDSocketProtocol
     {
+        ///////////////////////////////////////////////////////////////////////////
+        // internal interface
+
+	///\name Constructors
+	///@{
+
+        void init_client() const;       ///< Create unconnected client socket
+	                                /**< \note This member is implicitly called from the
+					     ProtocolClientSocketHandle::ProtocolClientSocketHandle()
+					     constructor */
+        void init_client(INet4Address const & address) const;
+                                        ///< Create client socket and connect
+                                        /**< Creates a new client socket and connects to the given
+					     address. 
+					     
+					     \param[in] address remote address to connect to */
+	                                /**< \note This member is implicitly called from the
+					     ProtocolClientSocketHandle::ProtocolClientSocketHandle()
+					     constructor */
+        void init_server() const;       ///< Create server socket
+	                                /**< \note This member is implicitly called from the
+					     ProtocolServerSocketHandle::ProtocolServerSocketHandle()
+					     constructor */
+        void init_server(INet4Address const & address, unsigned backlog=1) const;
+                                        ///< Create server socket and listen
+                                        /**< Creates a new server socket, binds to \a address end
+					     starts listening for new connections with a backlog of
+					     \a backlog connections. It also enables reuseaddr().
+
+					     \param[in] address address to listen on
+					     \param[in] backlog size of the listen backlog */
+	                                /**< \note This member is implicitly called from the
+					     ProtocolServerSocketHandle::ProtocolServerSocketHandle()
+					     constructor */
+
+	///@}
+	///\name Abstract Interface Implementation
+
+        std::auto_ptr<SocketProtocol> clone() const;
+	
+	///@}
     };
 
     typedef ProtocolClientSocketHandle<TCPv6SocketProtocol> TCPv6ClientSocketHandle;
diff --git a/doclib/doxy-header-overview.html b/doclib/doxy-header-overview.html
index e0f52b30f..0a43bb6c2 100644
--- a/doclib/doxy-header-overview.html
+++ b/doclib/doxy-header-overview.html
@@ -17,11 +17,9 @@ div.tabs ul li.$projectname a { background-color: #EDE497; }
 
 <div id="content1">
   <div id="content2">
-    <div class="tabs">
+    <div class="tabs menu">
       <ul>
         <li class="Overview"><a href="../../doc/html/index.html">Overview</a></li>
-        <li><a class="ext" href="http://developer.berlios.de/projects/senf">SENF @ BerliOS</a></li>
-
         <li class="libSocket"><a href="../../Socket/doc/html/index.html">libSocket</a></li>
         <li class="libPackets"><a href="../../Packets/doc/html/index.html">libPackets</a></li>
         <li class="libScheduler"><a href="../../Scheduler/doc/html/index.html">libScheduler</a></li>
@@ -29,3 +27,12 @@ div.tabs ul li.$projectname a { background-color: #EDE497; }
         <li class="SENFSCons"><a href="../../senfscons/doc/html/index.html">SENFSCons</a></li>
       </ul>
     </div>
+
+    <div class="tabs">
+      <ul>
+	<li><a href="xref.html">Open Issues</a></li>
+	<li><a class="ext" href="http://svn.berlios.de/wsvn/senf/?op=log&rev=0&sc=0&isdir=1">SVN ChangeLog</a></li>
+        <li><a class="ext" href="http://developer.berlios.de/projects/senf">SENF @ BerliOS</a></li>
+	<li><a class="ext" href="http://openfacts.berlios.de/index-en.phtml?title=SENF+Network+Framework">Wiki</a></li>
+      </ul>
+    </div>
\ No newline at end of file
diff --git a/doclib/doxy-header.html b/doclib/doxy-header.html
index a630f59ec..8eab86351 100644
--- a/doclib/doxy-header.html
+++ b/doclib/doxy-header.html
@@ -17,11 +17,9 @@ div.tabs ul li.$projectname a { background-color: #EDE497; }
 
 <div id="content1">
   <div id="content2">
-    <div class="tabs">
+    <div class="tabs menu">
       <ul>
         <li class="Overview"><a href="../../../doc/html/index.html">Overview</a></li>
-        <li><a class="ext" href="http://developer.berlios.de/projects/senf">SENF @ BerliOS</a></li>
-
         <li class="libSocket"><a href="../../../Socket/doc/html/index.html">libSocket</a></li>
         <li class="libPackets"><a href="../../../Packets/doc/html/index.html">libPackets</a></li>
         <li class="libScheduler"><a href="../../../Scheduler/doc/html/index.html">libScheduler</a></li>
diff --git a/doclib/filter.pl b/doclib/filter.pl
index eb9214db9..dd192c8ee 100755
--- a/doclib/filter.pl
+++ b/doclib/filter.pl
@@ -6,6 +6,9 @@ while (s/\t/' 'x(8-length($`)%8)/e) {}
 if (/^\s*\\code$/ .. /\\endcode/ && !/^$/) {
     $i=length($1) if /^(\s*)\\code$/;
     print substr($_,$i),"\n";
+} elsif (s/^(\s*)<pre>$/$1<pre class="fragment">/ .. /<\/pre>/ && !/^$/) {
+    $i=length($1) if /^(\s*)<pre class="fragment">$/;
+    print substr($_,$i),"\n";
 } else {
     print $_,"\n";
 }
diff --git a/doclib/senf.css b/doclib/senf.css
index 270f997f9..ad630ac07 100644
--- a/doclib/senf.css
+++ b/doclib/senf.css
@@ -14,24 +14,24 @@ body {
 
 #head h1 {
 	margin: 0 0 0 100px;
-        padding: 6px 0 0 0;
+	padding: 6px 0 0 42px;
 	height: 33px;
-        background-color: #DECD40;
+	background-color: #DECD40;
 	border-bottom: 1px solid #AF9D00;
-        font-size: 22px;
+	font-size: 22px;
 	font-weight: bold;
-        color: white;
+	color: white;
 	white-space: nowrap;
 	text-align: left;
 }
 
 #head h2 {
 	margin: 0 0 0 100px;
-	padding: 4px 0 0 0;
+	padding: 4px 0 0 42px;
 	height: 18px;
 	background-color: #EDE497;
 	color: #726921;
-        font-size: 13px;
+	font-size: 13px;
 	font-weight: normal;
 	white-space: nowrap;
 }
@@ -42,7 +42,7 @@ body {
 }
 
 #content2 { 
-        /* need non-zero top padding here to prevent margin propagation */
+	/* need non-zero top padding here to prevent margin propagation */
 	padding: 10px 0 0 142px;
 	max-width: 62em;
 }
@@ -57,7 +57,7 @@ a:contains("http://") {
 }
 
 a.ext {
-        font-style: italic;
+	font-style: italic;
 }
 
 div.tabs { 
@@ -66,7 +66,7 @@ div.tabs {
 	clear: left;
 	background-color: #FDF7C3;
 	border: 1px solid #AF9D00;
-        margin: 0 0 10px -132px;
+	margin: 0 0 10px -132px;
 	width: 120px; 
 	overflow: hidden;
 }
@@ -88,7 +88,7 @@ div.tabs ul li {
 div.tabs ul li a { 
 	display: block;
 	padding: 2px 5px;
-        font-size: 13px;
+	font-size: 13px;
 	color: #726921;
 	text-decoration: none;
 	white-space: nowrap;
@@ -97,17 +97,17 @@ div.tabs ul li a {
 
 div.tabs ul li a:visited, div.tabs ul li a:active { 
 	color: #726921;
-        text-decoration: none;
+	text-decoration: none;
 }
 
 div.tabs li.current a {
 	background-position: 100% -150px;
-	border-width     : 0px;
+	border-width: 0px;
 }
 
 div.tabs li.current span {
 	background-position: 0% -150px;
-	padding-bottom   : 6px;
+	padding-bottom: 6px;
 }
 
 div.tabs ul li a:hover, div.tabs ul li.current a { 
@@ -126,7 +126,7 @@ div.tabs ul li a:hover, div.tabs ul li.current a {
 }
 
 #footer span { 
-        font-size: 10px;
+	font-size: 10px;
 }
 
 #footer a, #footer a:active, #footer a:visited { 
-- 
GitLab