/* * Copyright (c) 2002, 2019, 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 4673103 * @library /test/lib * @run main/othervm/timeout=140 MarkResetTest * @summary URLConnection.getContent() hangs over FTP for DOC, PPT, XLS files */ import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.io.OutputStream; import java.io.PrintWriter; import java.io.UncheckedIOException; import java.net.InetAddress; import java.net.InetSocketAddress; import java.net.Proxy; import java.net.ServerSocket; import java.net.Socket; import java.net.URL; import java.net.URLConnection; import java.nio.file.Files; import java.nio.file.Paths; import jdk.test.lib.net.URIBuilder; public class MarkResetTest { private static final String FILE_NAME = "EncDec.doc"; /** * A class that simulates, on a separate, an FTP server. */ private class FtpServer extends Thread { private final ServerSocket server; private volatile boolean done = false; private boolean pasvEnabled = true; private boolean portEnabled = true; private boolean extendedEnabled = true; /** * This Inner class will handle ONE client at a time. * That's where 99% of the protocol handling is done. */ private class FtpServerHandler extends Thread { BufferedReader in; PrintWriter out; Socket client; private final int ERROR = 0; private final int USER = 1; private final int PASS = 2; private final int CWD = 3; private final int TYPE = 4; private final int RETR = 5; private final int PASV = 6; private final int PORT = 7; private final int QUIT = 8; private final int EPSV = 9; String[] cmds = { "USER", "PASS", "CWD", "TYPE", "RETR", "PASV", "PORT", "QUIT", "EPSV"}; private String arg = null; private ServerSocket pasv = null; private int data_port = 0; private InetAddress data_addr = null; /** * Parses a line to match it with one of the supported FTP commands. * Returns the command number. */ private int parseCmd(String cmd) { if (cmd == null || cmd.length() < 3) return ERROR; int blank = cmd.indexOf(' '); if (blank < 0) blank = cmd.length(); if (blank < 3) return ERROR; String s = cmd.substring(0, blank); if (cmd.length() > blank+1) arg = cmd.substring(blank+1, cmd.length()); else arg = null; for (int i = 0; i < cmds.length; i++) { if (s.equalsIgnoreCase(cmds[i])) return i+1; } return ERROR; } public FtpServerHandler(Socket cl) { client = cl; } protected boolean isPasvSet() { if (pasv != null && !pasvEnabled) { try { pasv.close(); } catch (IOException ex) { } pasv = null; } if (pasvEnabled && pasv != null) return true; return false; } /** * Open the data socket with the client. This can be the * result of a "PASV" or "PORT" command. */ protected OutputStream getOutDataStream() { try { if (isPasvSet()) { Socket s = pasv.accept(); return s.getOutputStream(); } if (data_addr != null) { Socket s = new Socket(data_addr, data_port); data_addr = null; data_port = 0; return s.getOutputStream(); } } catch (Exception e) { e.printStackTrace(); } return null; } protected InputStream getInDataStream() { try { if (isPasvSet()) { Socket s = pasv.accept(); return s.getInputStream(); } if (data_addr != null) { Socket s = new Socket(data_addr, data_port); data_addr = null; data_port = 0; return s.getInputStream(); } } catch (Exception e) { e.printStackTrace(); } return null; } /** * Handles the protocol exchange with the client. */ public void run() { done = false; String str; int res; boolean logged = false; boolean waitpass = false; try { in = new BufferedReader(new InputStreamReader( client.getInputStream())); out = new PrintWriter(client.getOutputStream(), true); out.println("220 tatooine FTP server (SunOS 5.8) ready."); } catch (Exception ex) { return; } while (!done) { try { str = in.readLine(); res = parseCmd(str); if ((res > PASS && res != QUIT) && !logged) { out.println("530 Not logged in."); continue; } switch (res) { case ERROR: out.println("500 '" + str + "': command not understood."); break; case USER: if (!logged && !waitpass) { out.println("331 Password required for " + arg); waitpass = true; } else { out.println("503 Bad sequence of commands."); } break; case PASS: if (!logged && waitpass) { out.println("230-Welcome to the FTP server!"); out.println("ab"); out.println("230 Guest login ok, " + "access restrictions apply."); logged = true; waitpass = false; } else out.println("503 Bad sequence of commands."); break; case QUIT: out.println("221 Goodbye."); out.flush(); out.close(); if (pasv != null) pasv.close(); done = true; break; case TYPE: out.println("200 Type set to " + arg + "."); break; case CWD: out.println("250 CWD command successful."); break; case EPSV: if (!extendedEnabled || !pasvEnabled) { out.println("500 EPSV is disabled, " + "use PORT instead."); continue; } if ("all".equalsIgnoreCase(arg)) { out.println("200 EPSV ALL command successful."); continue; } try { if (pasv == null) { pasv = new ServerSocket(); pasv.bind(new InetSocketAddress(server.getInetAddress(), 0)); } int port = pasv.getLocalPort(); out.println("229 Entering Extended" + " Passive Mode (|||" + port + "|)"); } catch (IOException ssex) { out.println("425 Can't build data connection:" + " Connection refused."); } break; case PASV: if (!pasvEnabled) { out.println("500 PASV is disabled, " + "use PORT instead."); continue; } try { if (pasv == null) { pasv = new ServerSocket(); pasv.bind(new InetSocketAddress("127.0.0.1", 0)); } int port = pasv.getLocalPort(); // Parenthesis are optional, so let's be // nasty and don't put them out.println("227 Entering Passive Mode" + " 127,0,0,1," + (port >> 8) + "," + (port & 0xff)); } catch (IOException ssex) { out.println("425 Can't build data connection:" + "Connection refused."); } break; case PORT: if (!portEnabled) { out.println("500 PORT is disabled, " + "use PASV instead"); continue; } StringBuffer host; int i = 0, j = 4; while (j > 0) { i = arg.indexOf(',', i + 1); if (i < 0) break; j--; } if (j != 0) { out.println("500 '" + arg + "':" + " command not understood."); continue; } try { host = new StringBuffer(arg.substring(0, i)); for (j = 0; j < host.length(); j++) if (host.charAt(j) == ',') host.setCharAt(j, '.'); String ports = arg.substring(i+1); i = ports.indexOf(','); data_port = Integer.parseInt( ports.substring(0, i)) << 8; data_port += (Integer.parseInt( ports.substring(i+1))); data_addr = InetAddress.getByName( host.toString()); out.println("200 Command okay."); } catch (Exception ex3) { data_port = 0; data_addr = null; out.println("500 '" + arg + "':" + " command not understood."); } break; case RETR: { File file = new File(arg); if (!file.exists()) { System.out.println("File not found"); out.println("200 Command okay."); out.println("550 '" + arg + "' No such file or directory."); break; } FileInputStream fin = new FileInputStream(file); OutputStream dout = getOutDataStream(); if (dout != null) { out.println("150 Binary data connection" + " for " + arg + " (" + client.getInetAddress(). getHostAddress() + ") (" + file.length() + " bytes)."); int c; int len = 0; while ((c = fin.read()) != -1) { dout.write(c); len++; } dout.flush(); dout.close(); fin.close(); out.println("226 Binary Transfer complete."); } else { out.println("425 Can't build data" + " connection: Connection refused."); } } break; } } catch (IOException ioe) { ioe.printStackTrace(); try { out.close(); } catch (Exception ex2) { } done = true; } } } } public FtpServer(int port) { this(InetAddress.getLoopbackAddress(), port); } public FtpServer(InetAddress address, int port) { try { if (address == null) { server = new ServerSocket(port); } else { server = new ServerSocket(); server.bind(new InetSocketAddress(address, port)); } } catch (IOException e) { throw new UncheckedIOException(e); } } public FtpServer() { this(null, 21); } public int getPort() { return server.getLocalPort(); } /** * A way to tell the server that it can stop. */ synchronized public void terminate() { done = true; } /* * All we got to do here is create a ServerSocket and wait for a * connection. When a connection happens, we just have to create * a thread that will handle it. */ public void run() { try { System.out.println("FTP server waiting for connections at: " + server.getLocalSocketAddress()); Socket client; client = server.accept(); (new FtpServerHandler(client)).start(); server.close(); } catch (Exception e) { } } } public static void main(String[] args) throws Exception { Files.copy(Paths.get(System.getProperty("test.src"), FILE_NAME), Paths.get(".", FILE_NAME)); new MarkResetTest(); } public MarkResetTest() { FtpServer server = null; try { server = new FtpServer(0); server.start(); int port = 0; while (port == 0) { Thread.sleep(500); port = server.getPort(); } URL url = URIBuilder.newBuilder() .scheme("ftp") .loopback() .port(port) .path("/" + FILE_NAME) .toURL(); URLConnection con = url.openConnection(Proxy.NO_PROXY); System.out.println("getContent: " + con.getContent()); System.out.println("getContent-length: " + con.getContentLength()); InputStream is = con.getInputStream(); /** * guessContentTypeFromStream method calls mark and reset methods * on the given stream. Make sure that calling * guessContentTypeFromStream repeatedly does not affect * reading from the stream afterwards */ System.out.println("Call GuessContentTypeFromStream()" + " several times.."); for (int i = 0; i < 5; i++) { System.out.println((i + 1) + " mime-type: " + con.guessContentTypeFromStream(is)); } int len = 0; int c; while ((c = is.read()) != -1) { len++; } is.close(); System.out.println("read: " + len + " bytes of the file"); // We're done! server.terminate(); server.interrupt(); // Did we pass ? if (len != (new File(FILE_NAME)).length()) { throw new Exception("Failed to read the file correctly"); } System.out.println("PASSED: File read correctly"); } catch (Exception e) { e.printStackTrace(); try { server.terminate(); server.interrupt(); } catch (Exception ex) { } throw new RuntimeException("FTP support error: " + e.getMessage()); } } }