All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.mycore.frontend.support.MCRSecureTokenV2 Maven / Gradle / Ivy

There is a newer version: 2024.05
Show newest version
/*
 * This file is part of ***  M y C o R e  ***
 * See http://www.mycore.de/ for details.
 *
 * MyCoRe 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 3 of the License, or
 * (at your option) any later version.
 *
 * MyCoRe 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 MyCoRe.  If not, see .
 */

package org.mycore.frontend.support;

import java.net.URI;
import java.net.URISyntaxException;
import java.nio.charset.StandardCharsets;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;
import java.util.Base64;
import java.util.Objects;
import java.util.stream.Collectors;
import java.util.stream.Stream;

/**
 * An implementation of SecureToken V2 used by "Wowza Streaming Engine".
 * 

* A description of the algorithm: *

*
    *
  1. A string is constructed by combining contentPath,'?' and all alphabetically sorted * parameters consisting of ipAddress, sharedSecret and any queryParameters
  2. *
  3. Generate an SHA-256 hash of that string builded in step 1 and {@link StandardCharsets#UTF_8}.
  4. *
  5. Generate a {@link Base64} encoded string of the digest of step 2.
  6. *
  7. replace '+' by '-' and '/' by '_' to make it a safe parameter * value.
  8. *
* * @author Thomas Scheffler (yagee) * @see JIRA Ticket MCR-1058 */ public class MCRSecureTokenV2 { private String contentPath, ipAddress, sharedSecret, hash; private String[] queryParameters; public MCRSecureTokenV2(String contentPath, String ipAddress, String sharedSecret, String... queryParameters) { this.contentPath = Objects.requireNonNull(contentPath, "'contentPath' may not be null"); this.ipAddress = Objects.requireNonNull(ipAddress, "'ipAddress' may not be null"); this.sharedSecret = Objects.requireNonNull(sharedSecret, "'sharedSecret' may not be null"); this.queryParameters = queryParameters; try { this.contentPath = new URI(null, null, this.contentPath, null).getRawPath(); } catch (URISyntaxException e) { throw new RuntimeException(e); } buildHash(); } private void buildHash() { String forHashing = Stream.concat(Stream.of(ipAddress, sharedSecret), Arrays.stream(queryParameters).filter(Objects::nonNull)) //case of HttpServletRequest.getQueryString()==null .sorted() .collect(Collectors.joining("&", contentPath + "?", "")); MessageDigest digest; try { digest = MessageDigest.getInstance("SHA-256"); } catch (NoSuchAlgorithmException e) { throw new RuntimeException(e);//should never happen for 'SHA-256' } digest.update(URI.create(forHashing).toASCIIString().getBytes(StandardCharsets.US_ASCII)); byte[] sha256 = digest.digest(); hash = Base64.getEncoder() .encodeToString(sha256) .chars() .map(x -> { switch (x) { case '+': return '-'; case '/': return '_'; default: return x; } }) .collect(StringBuilder::new, StringBuilder::appendCodePoint, StringBuilder::append) .toString(); } public String getHash() { return hash; } /** * Same as calling {@link #toURI(String, String, String)} with suffix="". */ public URI toURI(String baseURL, String hashParameterName) throws URISyntaxException { return toURI(baseURL, "", hashParameterName); } /** * Constructs an URL by using all information from the * {@link MCRSecureTokenV2#MCRSecureTokenV2(String, String, String, String...) constructor} except * ipAddress and sharedSecret and the supplied parameters. * * @param baseURL * a valid and absolute base URL * @param suffix * is appended to the contentPath * @param hashParameterName * the name of the query parameter that holds the hash value * @return an absolute URL consisting of all elements as stated above and queryParameters in the * given order appended by the hash parameter and the hash value from {@link #getHash()}. * @throws URISyntaxException if baseURL is not a valid URI */ public URI toURI(String baseURL, String suffix, String hashParameterName) throws URISyntaxException { Objects.requireNonNull(suffix, "'suffix' may not be null"); Objects.requireNonNull(hashParameterName, "'hashParameterName' may not be null"); if (hashParameterName.isEmpty()) { throw new IllegalArgumentException("'hashParameterName' may not be empty"); } URI context = new URI(baseURL); return context.resolve(Stream .concat(Arrays.stream(queryParameters).filter(Objects::nonNull), Stream.of(hashParameterName + "=" + hash)) .collect(Collectors.joining("&", baseURL + contentPath + suffix + "?", ""))); } @Override public int hashCode() { return getHash().hashCode(); } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } MCRSecureTokenV2 other = (MCRSecureTokenV2) obj; if (!hash.equals(other.hash)) { return false; } if (!contentPath.equals(other.contentPath)) { return false; } if (!ipAddress.equals(other.ipAddress)) { return false; } if (!sharedSecret.equals(other.sharedSecret)) { return false; } return Arrays.equals(queryParameters, other.queryParameters); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy