/**
* The MIT License (MIT)
*
* Copyright (c) 2014-2017 Yegor Bugayenko
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included
* in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
package org.takes.http;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.URI;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import lombok.EqualsAndHashCode;
import org.takes.Take;
/**
* Front remote control.
*
* <p>The class is immutable and thread-safe.
*
* @author Yegor Bugayenko (yegor256@gmail.com)
* @version $Id: c040168871faf48a995aa8641a1080f2decf68f1 $
* @since 0.1
* @checkstyle ClassDataAbstractionCouplingCheck (500 lines)
*/
@EqualsAndHashCode
public final class FtRemote implements Front {
/**
* Original front.
*/
private final Front origin;
/**
* Server socket.
*/
private final ServerSocket socket;
/**
* Indicates whether the front is secure (HTTPS).
*/
private final boolean secured;
/**
* Ctor.
* @param tks Take
* @throws IOException If fails
*/
public FtRemote(final Take tks) throws IOException {
this(new BkParallel(new BkBasic(tks)));
}
/**
* Ctor.
* @param bck Back
* @throws IOException If fails
*/
public FtRemote(final Back bck) throws IOException {
this(bck, FtRemote.random());
}
/**
* Ctor.
* @param bck Back
* @param skt Server socket to use
* @since 0.22
*/
public FtRemote(final Back bck, final ServerSocket skt) {
this(new FtBasic(bck, skt), skt, false);
}
/**
* Ctor.
* @param front Original front
* @param skt ServerSocket used
* @param sec Value of {@code true} if the front is secure,
* {@code false} otherwise
*/
FtRemote(final Front front, final ServerSocket skt,
final boolean sec) {
this.origin = front;
this.socket = skt;
this.secured = sec;
}
@Override
public void start(final Exit exit) throws IOException {
this.origin.start(exit);
}
/**
* Execute this script against a running front.
* @param script Script to run
* @throws IOException If fails
*/
public void exec(final FtRemote.Script script) throws IOException {
final AtomicBoolean exit = new AtomicBoolean();
final CountDownLatch latch = new CountDownLatch(1);
final Thread thread = new Thread(
new Runnable() {
@Override
public void run() {
try {
FtRemote.this.start(
new Exit() {
@Override
public boolean ready() {
latch.countDown();
return exit.get();
}
}
);
} catch (final IOException ex) {
throw new IllegalStateException(ex);
}
}
}
);
thread.start();
try {
// @checkstyle MagicNumber (1 line)
if (!latch.await(10L, TimeUnit.SECONDS)) {
throw new IllegalArgumentException(
"failed to start the app"
);
}
} catch (final InterruptedException ex) {
Thread.currentThread().interrupt();
throw new IllegalStateException(ex);
}
final String protocol;
if (this.secured) {
protocol = "https";
} else {
protocol = "http";
}
script.exec(
URI.create(
String.format(
"%s://localhost:%d",
protocol, this.socket.getLocalPort()
)
)
);
exit.set(true);
try {
thread.join();
} catch (final InterruptedException ex) {
Thread.currentThread().interrupt();
throw new IllegalStateException(ex);
}
}
/**
* Make a random socket.
* @return Socket
* @throws IOException If fails
*/
private static ServerSocket random() throws IOException {
final ServerSocket skt = new ServerSocket(0);
skt.setReuseAddress(true);
return skt;
}
/**
* Script to execute.
*/
public interface Script {
/**
* Execute it against this URI.
* @param home URI of the running front
* @throws IOException If fails
*/
void exec(URI home) throws IOException;
}
}