jdk-24/test/jdk/javax/security/auth/Subject/CurrentSubject.java

150 lines
5.6 KiB
Java

/*
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code 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
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
import javax.security.auth.Subject;
import java.security.AccessController;
import java.security.Principal;
import java.security.PrivilegedAction;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.atomic.AtomicInteger;
/*
* @test
* @bug 8267108
* @summary confirm current subject specification
* @run main/othervm
* -Djdk.security.auth.subject.useTL=false -Dtest=both CurrentSubject
* @run main/othervm
* -Djdk.security.auth.subject.useTL=true -Dtest=old CurrentSubject
* @run main/othervm
* -Djdk.security.auth.subject.useTL=true -Dtest=new CurrentSubject
*/
public class CurrentSubject {
static final boolean TEST_NEW = !System.getProperty("test").equals("old");
static final boolean TEST_OLD = !System.getProperty("test").equals("new");
static transient boolean failed = false;
static CountDownLatch cl = new CountDownLatch(1);
static AtomicInteger count = new AtomicInteger();
public static void main(String[] args) throws Exception {
// At the beginning, current subject is null
test("", null);
cl.await();
if (failed) {
throw new Exception("Failed");
}
}
/**
* Ensure the current subject is the expected Subject object.
*
* @param label label to print out
* @param expected the expected Subject
*/
synchronized static void check(String label, Subject expected) {
Subject cas = Subject.current();
Subject accs = Subject.getSubject(AccessController.getContext());
if (TEST_NEW && TEST_OLD && cas != accs) {
failed = true;
System.out.println(label + ": current " + s2s(cas)
+ " but getSubject is " + s2s(accs));
}
Subject interested = TEST_NEW ? cas : accs;
if (interested != expected) {
failed = true;
System.out.println(label + ": expected " + s2s(expected)
+ " but see " + s2s(interested));
} else {
System.out.println(label + ": " + s2s(expected));
}
}
/**
* Recursively testing on current subject with getAs() and thread creations.
*
* @param name the current label
* @param expected the expected Subject
*/
static Void test(String name, Subject expected) {
// Now it's the expected current subject
check(" ".repeat(name.length()) + "-> " + name, expected);
// Recursively check, do not go infinity
if (name.length() < 4) {
Subject another = new Subject();
another.getPrincipals().add(new RawPrincipal(name + "d"));
// run with a new subject, inside current subject will be the new subject
if (TEST_NEW) Subject.callAs(another, () -> test(name + 'c', another));
if (TEST_OLD) Subject.doAs(another, (PrivilegedAction<Void>) () -> test(name + 'd', another));
// run with null, inside current subject will be null
if (TEST_NEW) Subject.callAs(null, () -> test(name + 'C', null));
if (TEST_OLD) Subject.doAs(null, (PrivilegedAction<Void>) () -> test(name + 'D', null));
// new thread, inside current subject is unchanged
count.incrementAndGet();
new Thread(() -> {
try {
test(name + 't', expected);
try {
Thread.sleep(500);
} catch (Exception e) {
throw new AssertionError(e);
}
// by this time, parent thread should have exited the
// action and current subject reset, but here
// current subject unchanged.
test(name + 'T', expected);
} finally {
var n = count.decrementAndGet();
if (n == 0) {
cl.countDown();
}
assert n >= 0;
}
}).start();
}
// Now it's reset to original
check(" ".repeat(name.length()) + "<- " + name, expected);
return null;
}
static class RawPrincipal implements Principal {
String name;
RawPrincipal(String name) {
this.name = name;
}
@Override
public String getName() {
return name;
}
}
static String s2s(Subject s) {
return s == null ? null
: s.getPrincipals().iterator().next().getName();
}
}