org.jpac.ef.EfService Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of elbfisch.core Show documentation
Show all versions of elbfisch.core Show documentation
Open-source runtime system for component-based implementation of automation solutions with Java
The newest version!
/**
* PROJECT : Elbfisch - java process automation controller (jPac)
* MODULE : Service.java (versatile input output subsystem)
* VERSION : -
* DATE : -
* PURPOSE :
* AUTHOR : Bernd Schuster, MSK Gesellschaft fuer Automatisierung mbH, Schenefeld
* REMARKS : -
* CHANGES : CH#n
*
* This file is part of the jPac process automation controller.
* jPac 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 3 of the License, or
* (at your option) any later version.
*
* jPac 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 for more details.
*
* You should have received a copy of the GNU General Public License
* along with the jPac If not, see .
*/
package org.jpac.ef;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.FixedRecvByteBufAllocator;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.LengthFieldBasedFrameDecoder;
import io.netty.handler.codec.LengthFieldPrepender;
import io.netty.handler.ssl.SslContext;
import io.netty.handler.ssl.SslContextBuilder;
import io.netty.handler.ssl.util.SelfSignedCertificate;
import java.net.BindException;
import java.security.cert.CertificateException;
import javax.net.ssl.SSLException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
*
* @author berndschuster
*/
public class EfService implements Runnable{
public static final int DEFAULTPORT = 13685;
public static final int DEFAULTRECEIVEBUFFERSIZE = 32000;
private final Logger Log = LoggerFactory.getLogger("jpac.ef");
private SslContext sslCtx;
private ChannelFuture channelFuture;
private EventLoopGroup bossGroup;
private EventLoopGroup workerGroup;
private final boolean useSSL;
private final String bindAddress;
private final int port;
private final int receiveBufferSize;
public EfService(boolean useSSL, String bindAddress, int port, int receiveBufferSize) throws CertificateException, SSLException, InterruptedException{
this.useSSL = useSSL;
this.bindAddress = bindAddress;
this.port = port;
this.receiveBufferSize = receiveBufferSize;
}
@Override
public void run(){
try{
if (useSSL) {
SelfSignedCertificate ssc = new SelfSignedCertificate();
sslCtx = SslContextBuilder.forServer(ssc.certificate(), ssc.privateKey()).build();
} else {
sslCtx = null;
}
bossGroup = new NioEventLoopGroup();
workerGroup = new NioEventLoopGroup();
ServerBootstrap b = new ServerBootstrap();
b.group(bossGroup, workerGroup)
.channel(NioServerSocketChannel.class)
.childHandler(new ChannelInitializer() {
@Override
public void initChannel(SocketChannel ch) throws Exception {
ch.config().setRecvByteBufAllocator(new FixedRecvByteBufAllocator(receiveBufferSize));
ch.pipeline().addLast("frameDecoder", new LengthFieldBasedFrameDecoder(DEFAULTRECEIVEBUFFERSIZE, 0, 4, 0, 4));
ch.pipeline().addLast("frameEncoder", new LengthFieldPrepender(4));
ch.pipeline().addLast("handler", new CommandHandler(ch.remoteAddress()));
}
});
channelFuture = b.bind(bindAddress, port).sync();//TODO bind to more than one address
Log.info("Elbfisch communication server up and running (" + bindAddress + ":" + port + ")");
// Wait until the server socket is closed.
channelFuture.channel().closeFuture().sync();
} catch(Exception exc){
if (exc instanceof BindException) {
Log.error("Failed to start Elbfisch communication server: bind address already in use (" + bindAddress + ":" + port + ")");
} else {
Log.error("Failed to start Elbfisch communication server (" + bindAddress + ":" + port + ")", exc);
}
bossGroup.shutdownGracefully();
workerGroup.shutdownGracefully();
}
}
public void exchangeChangedSignals(){
synchronized(CommandHandler.getListOfActiveCommandHandlers()){
CommandHandler.getListOfActiveCommandHandlers().forEach((commandHandler)-> {
commandHandler.transferChangedClientOutputTransportsToSignals();
commandHandler.transferChangedSignalsToClientInputTransports();
});
}
}
public void start(){
Thread serviceStarter = new Thread(this);
serviceStarter.setName("Ef Service");
serviceStarter.start();
}
public void stop(){
try{
if (channelFuture != null) {
channelFuture.channel().close().sync();
}
}
catch(InterruptedException exc){/*ignore*/}
finally{
bossGroup.shutdownGracefully();
workerGroup.shutdownGracefully();
}
}
}