/** * JBoss, Home of Professional Open Source * Copyright Red Hat, Inc., and individual contributors. * * Licensed 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. */ package org.jboss.aerogear.unifiedpush.api.validation; import org.jboss.aerogear.unifiedpush.api.VariantType; import org.jboss.aerogear.unifiedpush.api.Installation; import javax.validation.ConstraintValidator; import javax.validation.ConstraintValidatorContext; import java.util.regex.Pattern; /** * Validator that will validate if the device token matches to {@code VariantType} specific pattern. * @see VariantType */ public class DeviceTokenValidator implements ConstraintValidator<DeviceTokenCheck, Installation> { /** * Pattern for iOS is pretty well defined as the library we use for sending assumes HEX. * @see <a href="https://github.com/notnoop/java-apns/blob/20c10ebd22e15a55c0c1c12695c535d37435dcfd/src/main/java/com/notnoop/apns/internal/Utilities.java#L114">notnoop apns</a> */ private static final Pattern IOS_DEVICE_TOKEN = Pattern.compile("(?i)[a-f0-9 -]{64,}"); /** * Pattern for android is harder to define that is why we kept it lenient it is at least 100 characters long and can * consist of digits, alphas, - and _ all have one of these separators */ private static final Pattern ANDROID_DEVICE_TOKEN = Pattern.compile("(?i)[0-9a-z\\-_:]{100,}"); /** * Pattern for windows is a uri that can be 1024 characters long * @see <a href="http://blogs.windows.com/windows_phone/b/wpdev/archive/2013/10/22/recommended-practices-for-using-microsoft-push-notification-service-mpns.aspx?Redirected=true">Windows developer blog</a> */ private static final Pattern WINDOWS_DEVICE_TOKEN = Pattern.compile("https?://.{0,1024}"); /** * The SimplePush token is an URI. While we strongly recommend https, it is in theory possible that users of the * AeroGear SimplePush Server do not protect the "update" endpoint via SSL. */ private static final Pattern SIMPLE_PUSH_DEVICE_TOKEN = Pattern.compile("https?://.{0,2000}"); /** * Pattern for Amazon is harder to define that is why we kept it lenient it is at least 100 characters long and can * consist of digits, alphas, - , _ and . and all have one of these separators */ private static final Pattern ADM_DEVICE_TOKEN = Pattern.compile("(?i)[0-9a-z\\-_.]{100,}"); @Override public void initialize(DeviceTokenCheck constraintAnnotation) { } @Override public boolean isValid(Installation installation, ConstraintValidatorContext context) { final String deviceToken = installation.getDeviceToken(); if (installation.getVariant() == null || installation.getVariant().getType() == null || deviceToken == null) { return false; } final VariantType type = installation.getVariant().getType(); return isValidDeviceTokenForVariant(deviceToken, type); } /** * Helper to run quick up-front validations. * * @param deviceToken the submitted device token * @param type type of the variant * @return true if the token is valid */ public static boolean isValidDeviceTokenForVariant(final String deviceToken, final VariantType type) { switch (type) { case IOS: return IOS_DEVICE_TOKEN.matcher(deviceToken).matches(); case ANDROID: return ANDROID_DEVICE_TOKEN.matcher(deviceToken).matches(); case WINDOWS_WNS: case WINDOWS_MPNS: return WINDOWS_DEVICE_TOKEN.matcher(deviceToken).matches(); case SIMPLE_PUSH: return SIMPLE_PUSH_DEVICE_TOKEN.matcher(deviceToken).matches(); case ADM: return ADM_DEVICE_TOKEN.matcher(deviceToken).matches(); } return false; } }