org.herodbx.scram100b2.common.message.ServerFirstMessage Maven / Gradle / Ivy
/*
 * Copyright 2017, OnGres.
 *
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
 * following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following
 * disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
 * following disclaimer in the documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 */
package org.herodbx.scram100b2.common.message;
import static org.herodbx.scram100b2.common.util.Preconditions.checkArgument;
import static org.herodbx.scram100b2.common.util.Preconditions.checkNotEmpty;
import org.herodbx.scram100b2.common.ScramAttributeValue;
import org.herodbx.scram100b2.common.ScramAttributes;
import org.herodbx.scram100b2.common.exception.ScramParseException;
import org.herodbx.scram100b2.common.util.StringWritable;
import org.herodbx.scram100b2.common.util.StringWritableCsv;
/**
 * Constructs and parses server-first-messages. Formal syntax is:
 *
 * {@code
 * server-first-message = [reserved-mext ","] nonce "," salt ","
 *                        iteration-count ["," extensions]
 * }
 *
 * Note that extensions are not supported.
 *
 * @see [RFC5802] Section 7
 */
public class ServerFirstMessage implements StringWritable {
    /**
     * Minimum allowed value for the iteration, as per the RFC.
     */
    public static final int ITERATION_MIN_VALUE = 4096;
    private final String clientNonce;
    private final String serverNonce;
    private final String salt;
    private final int iteration;
    /**
     * Constructs a server-first-message from a client-first-message and the additional required data.
     * @param clientNonce String representing the client-first-message
     * @param serverNonce Server serverNonce
     * @param salt The salt
     * @param iteration The iteration count (must be <= 4096)
     * @throws IllegalArgumentException If clientFirstMessage, serverNonce or salt are null or empty,
     *                                  or iteration < 4096
     */
    public ServerFirstMessage(
            String clientNonce, String serverNonce, String salt, int iteration
    ) throws IllegalArgumentException {
        this.clientNonce = checkNotEmpty(clientNonce, "clientNonce");
        this.serverNonce = checkNotEmpty(serverNonce, "serverNonce");
        this.salt = checkNotEmpty(salt, "salt");
        checkArgument(iteration >= ITERATION_MIN_VALUE, "iteration must be >= " + ITERATION_MIN_VALUE);
        this.iteration = iteration;
    }
    public String getClientNonce() {
        return clientNonce;
    }
    public String getServerNonce() {
        return serverNonce;
    }
    public String getNonce() {
        return clientNonce + serverNonce;
    }
    public String getSalt() {
        return salt;
    }
    public int getIteration() {
        return iteration;
    }
    @Override
    public StringBuffer writeTo(StringBuffer sb) {
        return StringWritableCsv.writeTo(
                sb,
                new ScramAttributeValue(ScramAttributes.NONCE, getNonce()),
                new ScramAttributeValue(ScramAttributes.SALT, salt),
                new ScramAttributeValue(ScramAttributes.ITERATION, iteration + "")
        );
    }
    /**
     * Parses a server-first-message from a String.
     * @param serverFirstMessage The string representing the server-first-message
     * @param clientNonce The serverNonce that is present in the client-first-message
     * @return The parsed instance
     * @throws ScramParseException If the argument is not a valid server-first-message
     * @throws IllegalArgumentException If either argument is empty or serverFirstMessage is not a valid message
     */
    public static ServerFirstMessage parseFrom(String serverFirstMessage, String clientNonce)
    throws ScramParseException, IllegalArgumentException {
        checkNotEmpty(serverFirstMessage, "serverFirstMessage");
        checkNotEmpty(clientNonce, "clientNonce");
        String[] attributeValues = StringWritableCsv.parseFrom(serverFirstMessage, 3, 0);
        if(attributeValues.length != 3) {
            throw new ScramParseException("Invalid server-first-message");
        }
        ScramAttributeValue nonce = ScramAttributeValue.parse(attributeValues[0]);
        if(ScramAttributes.NONCE.getChar() != nonce.getChar()) {
            throw new ScramParseException("serverNonce must be the 1st element of the server-first-message");
        }
        if(! nonce.getValue().startsWith(clientNonce)) {
            throw new ScramParseException("parsed serverNonce does not start with client serverNonce");
        }
        ScramAttributeValue salt = ScramAttributeValue.parse(attributeValues[1]);
        if(ScramAttributes.SALT.getChar() != salt.getChar()) {
            throw new ScramParseException("salt must be the 2nd element of the server-first-message");
        }
        ScramAttributeValue iteration = ScramAttributeValue.parse(attributeValues[2]);
        if(ScramAttributes.ITERATION.getChar() != iteration.getChar()) {
            throw new ScramParseException("iteration must be the 3rd element of the server-first-message");
        }
        int iterationInt;
        try {
            iterationInt = Integer.parseInt(iteration.getValue());
        } catch (NumberFormatException e) {
            throw new ScramParseException("invalid iteration");
        }
        return new ServerFirstMessage(
                clientNonce, nonce.getValue().substring(clientNonce.length()), salt.getValue(), iterationInt
        );
    }
    @Override
    public String toString() {
        return writeTo(new StringBuffer()).toString();
    }
}
    © 2015 - 2025 Weber Informatics LLC | Privacy Policy