package com.github.ltsopensource.core.commons.io; /* * SerialKiller.java * * Copyright (c) 2015 Luca Carettoni * * Easy to use library to secure Java deserialization from untrusted input. * * Dual-Licensed Software: * [Apache V2.0] * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the LICENSE 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. * [GPL V2.0] * This program 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 2 of the License, or * (at your option) any later version. This program is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY. * * @see https://github.com/ikkisoft/SerialKiller * @see http://www.infoq.com/cn/articles/java-deserialization-nsfocus */ import java.io.*; import java.util.regex.Matcher; import java.util.regex.Pattern; public class SerialKiller extends ObjectInputStream { private String[] blacklist; private String[] whitelist; public SerialKiller(InputStream inputStream) throws IOException { super(inputStream); blacklist = new String[]{ "org.apache.commons.collections.functors.InvokerTransformer$", "org.apache.commons.collections.functors.InstantiateTransformer$", "org.apache.commons.collections4.functors.InvokerTransformer$", "org.apache.commons.collections4.functors.InstantiateTransformer$", "org.codehaus.groovy.runtime.ConvertedClosure$", "org.codehaus.groovy.runtime.MethodClosure$", "org.springframework.beans.factory.ObjectFactory$" }; whitelist = new String[]{".*"}; } @Override protected Class<?> resolveClass(ObjectStreamClass serialInput) throws IOException, ClassNotFoundException { //Enforce SerialKiller's blacklist for (String blackRegExp : blacklist) { Pattern blackPattern = Pattern.compile(blackRegExp); Matcher blackMatcher = blackPattern.matcher(serialInput.getName()); if (blackMatcher.find()) { throw new InvalidClassException("[!] Blocked by SerialKiller's blacklist '" + blackRegExp + "'. Match found for '" + serialInput.getName() + "'"); } } //Enforce SerialKiller's whitelist boolean safeClass = false; for (String whiteRegExp : whitelist) { Pattern whitePattern = Pattern.compile(whiteRegExp); Matcher whiteMatcher = whitePattern.matcher(serialInput.getName()); if (whiteMatcher.find()) { safeClass = true; } } if (!safeClass) { throw new InvalidClassException("[!] Blocked by SerialKiller's whitelist. No match found for '" + serialInput.getName() + "'"); } return super.resolveClass(serialInput); } }