8262731: [macOS] Exception from "Printable.print" is swallowed during "PrinterJob.print"
Reviewed-by: prr
This commit is contained in:
parent
c4207355d3
commit
c0b4407d09
@ -33,6 +33,7 @@ import java.awt.print.*;
|
|||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
import java.security.AccessController;
|
import java.security.AccessController;
|
||||||
import java.security.PrivilegedAction;
|
import java.security.PrivilegedAction;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
import javax.print.*;
|
import javax.print.*;
|
||||||
import javax.print.attribute.PrintRequestAttributeSet;
|
import javax.print.attribute.PrintRequestAttributeSet;
|
||||||
@ -60,6 +61,7 @@ public final class CPrinterJob extends RasterPrinterJob {
|
|||||||
private static String sShouldNotReachHere = "Should not reach here.";
|
private static String sShouldNotReachHere = "Should not reach here.";
|
||||||
|
|
||||||
private volatile SecondaryLoop printingLoop;
|
private volatile SecondaryLoop printingLoop;
|
||||||
|
private AtomicReference<Throwable> printErrorRef = new AtomicReference<>();
|
||||||
|
|
||||||
private boolean noDefaultPrinter = false;
|
private boolean noDefaultPrinter = false;
|
||||||
|
|
||||||
@ -323,6 +325,7 @@ public final class CPrinterJob extends RasterPrinterJob {
|
|||||||
performingPrinting = true;
|
performingPrinting = true;
|
||||||
userCancelled = false;
|
userCancelled = false;
|
||||||
}
|
}
|
||||||
|
printErrorRef.set(null);
|
||||||
|
|
||||||
//Add support for PageRange
|
//Add support for PageRange
|
||||||
PageRanges pr = (attributes == null) ? null
|
PageRanges pr = (attributes == null) ? null
|
||||||
@ -381,6 +384,15 @@ public final class CPrinterJob extends RasterPrinterJob {
|
|||||||
if (printingLoop != null) {
|
if (printingLoop != null) {
|
||||||
printingLoop.exit();
|
printingLoop.exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Throwable printError = printErrorRef.getAndSet(null);
|
||||||
|
if (printError != null) {
|
||||||
|
if (printError instanceof PrinterException) {
|
||||||
|
throw (PrinterException) printError;
|
||||||
|
}
|
||||||
|
throw (PrinterException)
|
||||||
|
new PrinterException().initCause(printError);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Normalize the collated, # copies, numPages, first/last pages. Need to
|
// Normalize the collated, # copies, numPages, first/last pages. Need to
|
||||||
@ -786,22 +798,36 @@ public final class CPrinterJob extends RasterPrinterJob {
|
|||||||
private Rectangle2D printAndGetPageFormatArea(final Printable printable, final Graphics graphics, final PageFormat pageFormat, final int pageIndex) {
|
private Rectangle2D printAndGetPageFormatArea(final Printable printable, final Graphics graphics, final PageFormat pageFormat, final int pageIndex) {
|
||||||
final Rectangle2D[] ret = new Rectangle2D[1];
|
final Rectangle2D[] ret = new Rectangle2D[1];
|
||||||
|
|
||||||
Runnable r = new Runnable() { public void run() { synchronized(ret) {
|
Runnable r = new Runnable() {
|
||||||
try {
|
@Override
|
||||||
int pageResult = printable.print(graphics, pageFormat, pageIndex);
|
public void run() {
|
||||||
if (pageResult != Printable.NO_SUCH_PAGE) {
|
synchronized (ret) {
|
||||||
ret[0] = getPageFormatArea(pageFormat);
|
try {
|
||||||
|
int pageResult = printable.print(
|
||||||
|
graphics, pageFormat, pageIndex);
|
||||||
|
if (pageResult != Printable.NO_SUCH_PAGE) {
|
||||||
|
ret[0] = getPageFormatArea(pageFormat);
|
||||||
|
}
|
||||||
|
} catch (Throwable t) {
|
||||||
|
printErrorRef.compareAndSet(null, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (Exception e) {} // Original code bailed on any exception
|
}
|
||||||
}}};
|
};
|
||||||
|
|
||||||
if (onEventThread) {
|
if (onEventThread) {
|
||||||
try { EventQueue.invokeAndWait(r); } catch (Exception e) { e.printStackTrace(); }
|
try {
|
||||||
|
EventQueue.invokeAndWait(r);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
printErrorRef.compareAndSet(null, t);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
r.run();
|
r.run();
|
||||||
}
|
}
|
||||||
|
|
||||||
synchronized(ret) { return ret[0]; }
|
synchronized (ret) {
|
||||||
|
return ret[0];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// upcall from native
|
// upcall from native
|
||||||
|
@ -0,0 +1,134 @@
|
|||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* @test
|
||||||
|
@bug 8262731
|
||||||
|
@key headful printer
|
||||||
|
@summary Verify that "PrinterJob.print" throws the expected exception,
|
||||||
|
if "Printable.print" throws an exception.
|
||||||
|
@run main ExceptionFromPrintableIsIgnoredTest MAIN PE
|
||||||
|
@run main ExceptionFromPrintableIsIgnoredTest MAIN RE
|
||||||
|
@run main ExceptionFromPrintableIsIgnoredTest EDT PE
|
||||||
|
@run main ExceptionFromPrintableIsIgnoredTest EDT RE
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.awt.Graphics;
|
||||||
|
import java.awt.print.PageFormat;
|
||||||
|
import java.awt.print.Printable;
|
||||||
|
import java.awt.print.PrinterException;
|
||||||
|
import java.awt.print.PrinterJob;
|
||||||
|
import java.lang.reflect.InvocationTargetException;
|
||||||
|
import javax.swing.SwingUtilities;
|
||||||
|
|
||||||
|
public class ExceptionFromPrintableIsIgnoredTest {
|
||||||
|
private enum TestThreadType {MAIN, EDT}
|
||||||
|
private enum TestExceptionType {PE, RE}
|
||||||
|
|
||||||
|
private volatile Throwable printError;
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
if (args.length < 2) {
|
||||||
|
throw new RuntimeException("Two arguments are expected:"
|
||||||
|
+ " test thread type and test exception type.");
|
||||||
|
}
|
||||||
|
|
||||||
|
new ExceptionFromPrintableIsIgnoredTest(
|
||||||
|
TestThreadType.valueOf(args[0]),
|
||||||
|
TestExceptionType.valueOf(args[1]));
|
||||||
|
}
|
||||||
|
|
||||||
|
public ExceptionFromPrintableIsIgnoredTest(
|
||||||
|
final TestThreadType threadType,
|
||||||
|
final TestExceptionType exceptionType) {
|
||||||
|
System.out.println(String.format(
|
||||||
|
"Test started. threadType='%s', exceptionType='%s'",
|
||||||
|
threadType, exceptionType));
|
||||||
|
|
||||||
|
String osName = System.getProperty("os.name");
|
||||||
|
boolean isOSX = osName.toLowerCase().startsWith("mac");
|
||||||
|
if ((exceptionType == TestExceptionType.RE) && !isOSX) {
|
||||||
|
System.out.println(
|
||||||
|
"Currently this test scenario can be verified only on macOS.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
printError = null;
|
||||||
|
|
||||||
|
if (threadType == TestThreadType.MAIN) {
|
||||||
|
runTest(exceptionType);
|
||||||
|
} else if (threadType == TestThreadType.EDT) {
|
||||||
|
try {
|
||||||
|
SwingUtilities.invokeAndWait(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
runTest(exceptionType);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (InterruptedException | InvocationTargetException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (printError == null) {
|
||||||
|
throw new RuntimeException("No exception was thrown.");
|
||||||
|
} else if (!(printError instanceof PrinterException)) {
|
||||||
|
throw new RuntimeException("Unexpected exception was thrown.");
|
||||||
|
}
|
||||||
|
System.out.println("Test passed.");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void runTest(final TestExceptionType exceptionType) {
|
||||||
|
PrinterJob job = PrinterJob.getPrinterJob();
|
||||||
|
if (job.getPrintService() == null) {
|
||||||
|
System.out.println("No printers are available.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
job.setPrintable(new Printable() {
|
||||||
|
@Override
|
||||||
|
public int print(Graphics graphics, PageFormat pageFormat,
|
||||||
|
int pageIndex) throws PrinterException {
|
||||||
|
if (pageIndex > 1) {
|
||||||
|
return NO_SUCH_PAGE;
|
||||||
|
}
|
||||||
|
if (exceptionType == TestExceptionType.PE) {
|
||||||
|
throw new PrinterException(
|
||||||
|
"Exception from 'Printable.print'.");
|
||||||
|
} else if (exceptionType == TestExceptionType.RE) {
|
||||||
|
throw new RuntimeException(
|
||||||
|
"Exception from 'Printable.print'.");
|
||||||
|
}
|
||||||
|
return PAGE_EXISTS;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
job.print();
|
||||||
|
} catch (Throwable t) {
|
||||||
|
printError = t;
|
||||||
|
|
||||||
|
System.out.println("'PrinterJob.print' threw the exception:");
|
||||||
|
t.printStackTrace(System.out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user