From 22025b0c3a1d0f861df01b581fde74781884336c Mon Sep 17 00:00:00 2001
From: zzz <zzz@mail.i2p>
Date: Tue, 23 Apr 2013 18:22:48 +0000
Subject: [PATCH]  * Console: Fix Jetty digest auth bug causing repeated
 password requests    I2P fixes for out-of-order nonce counts.    Based on
 DigestAuthenticator in Jetty 7.6.10.    Includes the nonce count verification
 code from Tomcat 7.0.35.    ref: http://jira.codehaus.org/browse/JETTY-1468
 which was closed not-a-bug.    ref:
 https://bugs.eclipse.org/bugs/show_bug.cgi?id=336443 in which the    Jetty
 implementation was introduced.

---
 apps/jetty/build.xml                          |   3 +-
 .../net/i2p/jetty/I2PDigestAuthenticator.java | 126 ++++++++++++------
 .../i2p/router/web/RouterConsoleRunner.java   |   7 +-
 3 files changed, 95 insertions(+), 41 deletions(-)

diff --git a/apps/jetty/build.xml b/apps/jetty/build.xml
index 97bb3abe1a..940d82e6e7 100644
--- a/apps/jetty/build.xml
+++ b/apps/jetty/build.xml
@@ -202,6 +202,7 @@
                 <pathelement location="./jettylib/javax.servlet.jar" />
                 <pathelement location="./jettylib/jetty-http.jar" />
                 <pathelement location="./jettylib/jetty-io.jar" />
+                <pathelement location="./jettylib/jetty-security.jar" />
                 <pathelement location="./jettylib/jetty-util.jar" />
                 <pathelement location="./jettylib/jetty-xml.jar" />
             </classpath>
@@ -216,7 +217,7 @@
             debug="true" deprecation="on" source="1.5" target="1.5" 
             destdir="./build/obj" 
             includeAntRuntime="false"
-            classpath="../../core/java/build/i2p.jar:./jettylib/commons-logging.jar:./jettylib/javax.servlet.jar:./jettylib/org.mortbay.jetty.jar:./jettylib/jetty-http.jar:./jettylib/jetty-io.jar:./jettylib/jetty-util.jar:./jettylib/jetty-xml.jar" >
+            classpath="../../core/java/build/i2p.jar:./jettylib/commons-logging.jar:./jettylib/javax.servlet.jar:./jettylib/org.mortbay.jetty.jar:./jettylib/jetty-http.jar:./jettylib/jetty-io.jar:./jettylib/jetty-security.jar:./jettylib/jetty-util.jar:./jettylib/jetty-xml.jar" >
             <compilerarg line="${javac.compilerargs}" />
         </javac>
     </target>
diff --git a/apps/jetty/java/src/net/i2p/jetty/I2PDigestAuthenticator.java b/apps/jetty/java/src/net/i2p/jetty/I2PDigestAuthenticator.java
index 8b58826944..f749effe1b 100644
--- a/apps/jetty/java/src/net/i2p/jetty/I2PDigestAuthenticator.java
+++ b/apps/jetty/java/src/net/i2p/jetty/I2PDigestAuthenticator.java
@@ -16,7 +16,7 @@
 //  ========================================================================
 //
 
-package org.eclipse.jetty.security.authentication;
+package net.i2p.jetty;
 
 import java.io.IOException;
 import java.security.MessageDigest;
@@ -25,7 +25,6 @@ import java.util.Queue;
 import java.util.concurrent.ConcurrentHashMap;
 import java.util.concurrent.ConcurrentLinkedQueue;
 import java.util.concurrent.ConcurrentMap;
-import java.util.concurrent.atomic.AtomicInteger;
 
 import javax.servlet.ServletRequest;
 import javax.servlet.ServletResponse;
@@ -36,6 +35,8 @@ import org.eclipse.jetty.http.HttpHeaders;
 import org.eclipse.jetty.security.SecurityHandler;
 import org.eclipse.jetty.security.ServerAuthException;
 import org.eclipse.jetty.security.UserAuthentication;
+import org.eclipse.jetty.security.authentication.DeferredAuthentication;
+import org.eclipse.jetty.security.authentication.DigestAuthenticator;
 import org.eclipse.jetty.server.Authentication;
 import org.eclipse.jetty.server.Authentication.User;
 import org.eclipse.jetty.server.Request;
@@ -50,74 +51,110 @@ import org.eclipse.jetty.util.security.Constraint;
 import org.eclipse.jetty.util.security.Credential;
 
 /**
+ * I2P fixes for out-of-order nonce counts.
+ * Based on DigestAuthenticator in Jetty 7.6.10.
+ * Includes the nonce count verification code from Tomcat 7.0.35.
+ * ref: http://jira.codehaus.org/browse/JETTY-1468 which was closed not-a-bug.
+ * ref: https://bugs.eclipse.org/bugs/show_bug.cgi?id=336443 in which the
+ * Jetty implementation was introduced.
+ * 
+ * @since 0.9.6
+ * 
  * @version $Rev: 4793 $ $Date: 2009-03-19 00:00:01 +0100 (Thu, 19 Mar 2009) $
  * 
  * The nonce max age in ms can be set with the {@link SecurityHandler#setInitParameter(String, String)} 
  * using the name "maxNonceAge"
  */
-public class DigestAuthenticator extends LoginAuthenticator
+public class I2PDigestAuthenticator extends DigestAuthenticator
 {
-    private static final Logger LOG = Log.getLogger(DigestAuthenticator.class);
+    // shadows super
+    private static final Logger LOG = Log.getLogger(I2PDigestAuthenticator.class);
     SecureRandom _random = new SecureRandom();
-    private long _maxNonceAgeMs = 60*1000;
+    // shadows super
+    private long _maxNonceAgeMs = 60*60*1000L;
     private ConcurrentMap<String, Nonce> _nonceCount = new ConcurrentHashMap<String, Nonce>();
+    // shadows super
     private Queue<Nonce> _nonceQueue = new ConcurrentLinkedQueue<Nonce>();
+
+    /*
+     * Shadows super
+     *
+     * Contains code from Tomcat 7.0.35 DigestAuthenticator.NonceInfo
+     *
+     * Licensed to the Apache Software Foundation (ASF) under one or more
+     * contributor license agreements.  See the NOTICE file distributed with
+     * this work for additional information regarding copyright ownership.
+     * The ASF licenses this file to You under the Apache License, Version 2.0
+     * (the "License"); you may not use this file except in compliance with
+     * the License.  You may obtain a copy of the License at
+     *
+     *      http://www.apache.org/licenses/LICENSE-2.0
+     *
+     * Unless required by applicable law or agreed to in writing, software
+     * distributed under the License is distributed on an "AS IS" BASIS,
+     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+     * See the License for the specific language governing permissions and
+     * limitations under the License.
+     */
     private static class Nonce
     {
         final String _nonce;
         final long _ts;
-        AtomicInteger _nc=new AtomicInteger();
+        private volatile boolean seen[];
+        private volatile int offset;
+        private volatile int count = 0;
+        private static final int seenWindowSize = 100;
+
         public Nonce(String nonce, long ts)
         {
             _nonce=nonce;
             _ts=ts;
+            seen = new boolean[seenWindowSize];
+            offset = seenWindowSize / 2;
+        }
+
+        public synchronized boolean nonceCountValid(long nonceCount) {
+            if ((count - offset) >= nonceCount ||
+                    (nonceCount > count - offset + seen.length)) {
+                return false;
+            }
+            int checkIndex = (int) ((nonceCount + offset) % seen.length);
+            if (seen[checkIndex]) {
+                return false;
+            } else {
+                seen[checkIndex] = true;
+                seen[count % seen.length] = false;
+                count++;
+                return true;
+            }
         }
     }
 
     /* ------------------------------------------------------------ */
-    public DigestAuthenticator()
+    public I2PDigestAuthenticator()
     {
         super();
     }
 
+    
     /* ------------------------------------------------------------ */
+
     /**
-     * @see org.eclipse.jetty.security.authentication.LoginAuthenticator#setConfiguration(org.eclipse.jetty.security.Authenticator.AuthConfiguration)
+     *  Store local copy since field in super is private
      */
     @Override
-    public void setConfiguration(AuthConfiguration configuration)
-    {
-        super.setConfiguration(configuration);
-        
-        String mna=configuration.getInitParameter("maxNonceAge");
-        if (mna!=null)
-        {
-            synchronized (this)
-            {
-                _maxNonceAgeMs=Long.valueOf(mna);
-            }
-        }
-    }
-    
-    /* ------------------------------------------------------------ */
     public synchronized void setMaxNonceAge(long maxNonceAgeInMillis)
     {
+        super.setMaxNonceAge(maxNonceAgeInMillis);
         _maxNonceAgeMs = maxNonceAgeInMillis;
     }
 
     /* ------------------------------------------------------------ */
-    public String getAuthMethod()
-    {
-        return Constraint.__DIGEST_AUTH;
-    }
-
-    /* ------------------------------------------------------------ */
-    public boolean secureResponse(ServletRequest req, ServletResponse res, boolean mandatory, User validatedUser) throws ServerAuthException
-    {
-        return true;
-    }
 
-    /* ------------------------------------------------------------ */
+    /**
+     *  No changes from super
+     */
+    @Override
     public Authentication validateRequest(ServletRequest req, ServletResponse res, boolean mandatory) throws ServerAuthException
     {
         if (!mandatory)
@@ -224,6 +261,11 @@ public class DigestAuthenticator extends LoginAuthenticator
     }
 
     /* ------------------------------------------------------------ */
+
+    /**
+     *  No changes from super
+     */
+    @Override
     public String newNonce(Request request)
     {
         Nonce nonce;
@@ -247,6 +289,10 @@ public class DigestAuthenticator extends LoginAuthenticator
      * @return -1 for a bad nonce, 0 for a stale none, 1 for a good nonce
      */
     /* ------------------------------------------------------------ */
+
+    /**
+     *  Contains fixes
+     */
     private int checkNonce(Digest digest, Request request)
     {
         // firstly let's expire old nonces
@@ -274,12 +320,9 @@ public class DigestAuthenticator extends LoginAuthenticator
             long count = Long.parseLong(digest.nc,16);
             if (count>Integer.MAX_VALUE)
                 return 0;
-            int old=nonce._nc.get();
-            while (!nonce._nc.compareAndSet(old,(int)count))
-                old=nonce._nc.get();
-            if (count<=old)
+            if (!nonce.nonceCountValid(count)) {
                 return -1;
- 
+            }
             return 1;
         }
         catch (Exception e)
@@ -292,6 +335,11 @@ public class DigestAuthenticator extends LoginAuthenticator
     /* ------------------------------------------------------------ */
     /* ------------------------------------------------------------ */
     /* ------------------------------------------------------------ */
+
+    /**
+     *  Shadows super.
+     *  No changes from super
+     */
     private static class Digest extends Credential
     {
         private static final long serialVersionUID = -2484639019549527724L;
diff --git a/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java b/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java
index dd481d48bd..069819963a 100644
--- a/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java
+++ b/apps/routerconsole/java/src/net/i2p/router/web/RouterConsoleRunner.java
@@ -29,6 +29,7 @@ import static net.i2p.app.ClientAppState.*;
 import net.i2p.apps.systray.SysTray;
 import net.i2p.data.Base32;
 import net.i2p.data.DataHelper;
+import net.i2p.jetty.I2PDigestAuthenticator;
 import net.i2p.jetty.I2PLogger;
 import net.i2p.router.RouterContext;
 import net.i2p.router.update.ConsoleUpdateManager;
@@ -105,7 +106,11 @@ public class RouterConsoleRunner implements RouterApp {
     private static final String DEFAULT_WEBAPP_CONFIG_FILENAME = "webapps.config";
 
     // Jetty Auth
-    private static final DigestAuthenticator authenticator = new DigestAuthenticator();
+    private static final DigestAuthenticator authenticator = new I2PDigestAuthenticator();
+    static {
+        // default changed from 0 (forever) in Jetty 6 to 60*1000 ms in Jetty 7
+        authenticator.setMaxNonceAge(7*24*60*60*1000L);
+    }
     public static final String JETTY_REALM = "i2prouter";
     private static final String JETTY_ROLE = "routerAdmin";
     public static final String PROP_CONSOLE_PW = "routerconsole.auth." + JETTY_REALM;
-- 
GitLab