package dev.peerat.framework; import java.io.BufferedReader; import java.io.InputStream; import java.io.InputStreamReader; import java.lang.reflect.Constructor; import java.lang.reflect.Method; import java.net.ServerSocket; import java.net.Socket; import java.security.Key; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.function.Consumer; import java.util.function.Function; import javax.net.ssl.SSLServerSocket; import javax.net.ssl.SSLServerSocketFactory; import org.jose4j.jwa.AlgorithmConstraints.ConstraintType; import org.jose4j.jwk.JsonWebKey.OutputControlLevel; import org.jose4j.jwk.RsaJsonWebKey; import org.jose4j.jwk.RsaJwkGenerator; import org.jose4j.jws.AlgorithmIdentifiers; import org.jose4j.jws.JsonWebSignature; import org.jose4j.jwt.JwtClaims; import org.jose4j.jwt.consumer.InvalidJwtException; import org.jose4j.jwt.consumer.JwtConsumer; import org.jose4j.jwt.consumer.JwtConsumerBuilder; import org.jose4j.lang.JoseException; public class Router{ public static void main(String[] args){ } private Locker logger; private Locker exceptions; private RouteMapper[] mappers; private List interceptors; private Response noFileFound; private RsaJsonWebKey rsaJsonWebKey; private JwtConsumer jwtConsumer; private Consumer claims; private Function userCreator; private String[][] headers; private ServerSocket serverSocket; public Router(){ this.logger = new Locker<>(); this.exceptions = new Locker<>(); int types = RequestType.values().length; this.mappers = new RouteMapper[types]; this.interceptors = new ArrayList<>(); for(RequestType type : RequestType.values()) this.mappers[type.ordinal()] = new RouteMapper<>(this); this.headers = new String[types][0]; } public Router configureJwt(Consumer consumer, Consumer claims, Function userCreator) throws Exception{ this.rsaJsonWebKey = RsaJwkGenerator.generateJwk(2048); configureJwtWithKey(consumer, claims, userCreator, this.rsaJsonWebKey.getKey()); return this; } public Router configureJwt(Consumer consumer, Consumer claims, Function userCreator, Map keyParams) throws Exception{ this.rsaJsonWebKey = new RsaJsonWebKey(keyParams); configureJwtWithKey(consumer, claims, userCreator, this.rsaJsonWebKey.getKey()); return this; } private void configureJwtWithKey(Consumer consumer, Consumer claims, Function userCreator, Key key) throws Exception{ JwtConsumerBuilder builder = new JwtConsumerBuilder() .setRequireExpirationTime() .setAllowedClockSkewInSeconds(30) .setVerificationKey(key) .setJwsAlgorithmConstraints(ConstraintType.PERMIT, AlgorithmIdentifiers.RSA_USING_SHA256); consumer.accept(builder); this.jwtConsumer = builder.build(); this.claims = claims; this.userCreator = userCreator; } public Map exportJwtKey(){ return this.rsaJsonWebKey.toParams(OutputControlLevel.INCLUDE_PRIVATE); } public Router addDefaultHeaders(RequestType type, String... headers){ String[] origin = this.headers[type.ordinal()]; String[] copy = new String[origin.length+headers.length]; System.arraycopy(origin, 0, copy, 0, origin.length); System.arraycopy(headers, 0, copy, origin.length, headers.length); this.headers[type.ordinal()] = copy; return this; } public Router register(Response response){ try{ Method method = response.getClass().getDeclaredMethod("exec", Response.class.getDeclaredMethods()[0].getParameterTypes()); Route route = method.getAnnotation(Route.class); this.mappers[route.type().ordinal()].register(response, method, route); System.out.println("Registered route "+method+" ["+route.type()+" "+route.path()+"]"); }catch(Exception e){ throw new IllegalArgumentException(e); } return this; } public Router registerClass(Class clazz, Object... injections) throws Exception{ return registerClass(clazz, new DependencyInjector().of(injections)); } public Router registerClass(Class clazz, DependencyInjector injector) throws Exception{ Object instance = null; for(Method method : clazz.getDeclaredMethods()){ Route route = method.getAnnotation(Route.class); if(route == null) continue; if(instance == null){ Constructor[] constructors = clazz.getDeclaredConstructors(); if(constructors.length != 1) throw new IllegalArgumentException("Class with multiple constructor. Not supported by this framework for the moment."); instance = constructors[0].newInstance(injector.apply(constructors[0])); } this.mappers[route.type().ordinal()].register(instance, method, route); System.out.println("Registered route "+method+" ["+route.type()+" "+route.path()+"]"); } return this; } public Router registerPackages(Object... injections){ return registerPackages(new DependencyInjector().of(injections)); } public Router registerPackages(DependencyInjector injector){ String clazz = Thread.currentThread().getStackTrace()[2].getClassName(); String pack = clazz.substring(0, clazz.lastIndexOf('.')); return registerPackages(pack, injector); } public Router registerPackages(String name, Object... injections){ return registerPackages(name, new DependencyInjector().of(injections)); } public Router registerPackages(String name, DependencyInjector injector){ InputStream stream = ClassLoader.getSystemClassLoader().getResourceAsStream(name.replace(".", "/")); try{ BufferedReader reader = new BufferedReader(new InputStreamReader(stream)); String line; while((line = reader.readLine()) != null){ if(line.endsWith(".class")){ Class clazz = Class.forName(name+"."+line.substring(0, line.length()-6)); registerClass(clazz, injector); continue; } if(!line.contains(".")) registerPackages(name+"."+line, injector); } reader.close(); }catch(Exception e){ System.err.println("Failed to read "+name); e.printStackTrace(); } return this; } public Router setDefault(Response response){ this.noFileFound = response; return this; } public Router addInterceptor(RouteInterceptor interceptor){ this.interceptors.add(interceptor); return this; } public Router activeReOrdering(){ for(RouteMapper mapper : this.mappers) mapper.activeReOrdering(); return this; } String[] getDefaultHeaders(RequestType type){ return this.headers[type.ordinal()]; } void exec(Context context, HttpReader reader, HttpWriter writer) throws Exception{ if(this.mappers[context.getType().ordinal()].exec(context, reader, writer, this.interceptors)) return; if(noFileFound != null) noFileFound.exec(null, context, reader, writer); } public Router configureSSL(String keyStore, String keyStorePassword){ System.setProperty("javax.net.ssl.keyStore", keyStore); System.setProperty("javax.net.ssl.keyStorePassword", keyStorePassword); return this; } public U getUser(String token) throws InvalidJwtException{ return this.userCreator.apply(this.jwtConsumer.processToClaims(token)); } public String createAuthUser(U user) throws JoseException{ JwtClaims claims = new JwtClaims(); claims.setGeneratedJwtId(); // a unique identifier for the token claims.setIssuedAtToNow(); // when the token was issued/created (now) claims.setNotBeforeMinutesInThePast(2); // time before which the token is not yet valid (2 minutes ago) this.claims.accept(claims); user.write(claims); JsonWebSignature jws = new JsonWebSignature(); jws.setPayload(claims.toJson()); jws.setKey(rsaJsonWebKey.getPrivateKey()); jws.setKeyIdHeaderValue(rsaJsonWebKey.getKeyId()); jws.setAlgorithmHeaderValue(AlgorithmIdentifiers.RSA_USING_SHA256); return jws.getCompactSerialization(); } public Locker getLogger(){ return this.logger; } public Locker getExceptionLogger(){ return this.exceptions; } public void listen(int port, boolean ssl) throws Exception{ if (ssl) { // Not needed with the use of a proxy try { SSLServerSocketFactory ssf = (SSLServerSocketFactory) SSLServerSocketFactory.getDefault(); serverSocket = (SSLServerSocket) ssf.createServerSocket(port); while (!serverSocket.isClosed()) { Socket socket = serverSocket.accept(); Client client = new Client<>(socket, this); client.start(); } } catch (Exception e) { e.printStackTrace(); } finally { stop(); } } else { try{ serverSocket = new ServerSocket(port); while (!serverSocket.isClosed()) { Socket socket = serverSocket.accept(); Client client = new Client<>(socket, this); client.start(); } } catch (Exception e) { e.printStackTrace(); } finally { stop(); } } } public void stop(){ if(serverSocket == null) return; try { serverSocket.close(); serverSocket = null; }catch(Exception e){ e.printStackTrace(); } } }