view src/luan/lib/webserver/handlers/DomainHandler.java @ 1401:ef1620aa99cb

fix gc issues
author Franklin Schmidt <fschmidt@gmail.com>
date Mon, 16 Sep 2019 22:51:41 -0400
parents 221eedb0f54e
children
line wrap: on
line source

package luan.lib.webserver.handlers;

import java.io.Closeable;
import java.io.IOException;
import java.lang.ref.Reference;
//import java.lang.ref.WeakReference;
import java.lang.ref.SoftReference;
import java.lang.ref.ReferenceQueue;
import java.util.Map;
import java.util.HashMap;
import luan.lib.logging.Logger;
import luan.lib.logging.LoggerFactory;
import luan.lib.webserver.Handler;
import luan.lib.webserver.Request;
import luan.lib.webserver.Response;


public final class DomainHandler implements Handler {
	private static final Logger logger = LoggerFactory.getLogger(DomainHandler.class);

	public interface Factory {
		public Handler newHandler(String domain);
	}

	private static void close(Handler handler) {
		if( handler instanceof Closeable ) {
			try {
				((Closeable)handler).close();
			} catch(IOException e) {
				logger.error(handler.toString(),e);
			}
		}
	}

	private final Map<String,Reference<Handler>> map = new HashMap<String,Reference<Handler>>();

	private final Factory factory;

	public DomainHandler(Factory factory) {
		this.factory = factory;
	}

	public Response handle(Request request) {
		String host = (String)request.headers.get("host");
		if( host == null )
			return null;
		int i = host.indexOf(':');
		String domain = i == -1 ? host : host.substring(0,i);
		Handler handler = getHandler(domain);
		return handler==null ? null : handler.handle(request);
	}

	public Handler getHandler(String domain) {
		domain = domain.toLowerCase();
		synchronized(map) {
			Reference<Handler> ref = map.get(domain);
			Handler handler = ref==null ? null : ref.get();
			if( handler == null ) {
				//if(ref!=null) logger.info("gc "+domain);
				handler = factory.newHandler(domain);
				if( handler == null )
					return null;
				map.put(domain,new SoftReference<Handler>(handler));
			}
			return handler;
		}
	}

	public void removeHandler(String domain) {
		logger.info("removeHandler "+domain);
		domain = domain.toLowerCase();
		synchronized(map) {
			Reference<Handler> ref = map.remove(domain);
			Handler handler = ref==null ? null : ref.get();
			if( handler != null ) {
				close(handler);
			}
		}
	}

}