/*
 * GrowlNetwork.java Copyright (C) 2019. Daniel H. Huson
 *
 *  (Some files contain contributions from other authors, who are then mentioned separately.)
 *
 *  This program 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.
 *
 *  This program 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 this program.  If not, see <http://www.gnu.org/licenses/>.
 */

package jloda.util;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.nio.ByteBuffer;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;

/**
 * Interacts with Growl by using the socket service.
 *
 * @author chamerling
 */
public class GrowlNetwork {
    public static final int DEFAULT_PORT = 9887;
    public static final String DEFAULT_HOST = "localhost";
    private static final byte TYPE_REGISTRATION = 0;
    private static final byte TYPE_NOTIFICATION = 1;
    private static final String NOTIFICATION_NAME = "JavaGrowler";
    private final byte PROTOCOL_VERSION = 1;

    private String host;
    private int port;

    private static String appName = "GrowlNetwork";

    private static GrowlNetwork instance = null;

    /**
     * gets the instance of the GrowlNetwork object
     *
     * @return instance
     */
    public static GrowlNetwork getInstance() {
        if (instance == null) {
            String name = ProgramProperties.getProgramName();
            if (name != null && name.length() > 0)
                appName = name;
            return getInstance(appName, "");
        }
        return instance;
    }

    /**
     * gets an instance of the GrowlNetwork object
     *
     * @return instance
     */
    public static GrowlNetwork getInstance(String name, String passwd) {
        if (instance == null) {
            instance = GrowlNetwork.register(name, passwd);
        }
        return instance;
    }

    /**
     * notify
     *
     * @param title
     * @param message
     */
    public static void notify(String title, String message) {
        GrowlNetwork g = GrowlNetwork.getInstance();
        g.notify(appName, title, message, "");
    }

    /**
     * @param host
     * @param port
     */
    private GrowlNetwork(String host, int port) {
        this.host = host;
        this.port = port;
    }

    /*
     * (non-Javadoc)
     *
     * @see org.chamerling.javagrowl.Growl#notify(java.lang.String,
     * java.lang.String, java.lang.String, java.lang.String)
     */
    public void notify(String appName, String title, String message, String password) {
        sendPacket(notificationPacket(appName, title, message, password).array());
    }

    private byte[] stringEnc(String str) {
        try {
            return str.getBytes("UTF8");
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
        return new byte[0];
    }

    private byte[] md5(byte[] bytes, String password) {
        MessageDigest md;
        try {
            md = MessageDigest.getInstance("MD5");
            md.update(bytes);

            if (password != null && password.length() > 0) {
                md.update(stringEnc(password));
            }
            return md.digest();
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        }
        return null;
    }

    private ByteBuffer registrationPacket(String appName, String password) {
        byte[] name = stringEnc(appName);
        byte[] notName = stringEnc(NOTIFICATION_NAME);

        int len = 6 + name.length + 2 + notName.length + 1 + 16;

        ByteBuffer bb = ByteBuffer.allocate(len);
        bb.put(PROTOCOL_VERSION);
        bb.put(TYPE_REGISTRATION);
        bb.putShort((short) name.length);
        bb.put((byte) 1); // nall
        bb.put((byte) 1); // ndef
        bb.put(name);
        bb.putShort((short) notName.length);
        bb.put(notName);
        bb.put((byte) 0); // defaults
        bb.put(md5(Arrays.copyOf(bb.array(), len - 16), password));
        return bb;
    }

    private ByteBuffer notificationPacket(String appName, String title, String message, String password) {
        byte[] uappName = stringEnc(appName);
        byte[] unotif = stringEnc(NOTIFICATION_NAME);
        byte[] utitle = stringEnc(title);
        byte[] umessage = stringEnc(message);

        int len = 12 + unotif.length + utitle.length + umessage.length + uappName.length + 16;
        ByteBuffer bb = ByteBuffer.allocate(len);

        bb.put(PROTOCOL_VERSION);
        bb.put(TYPE_NOTIFICATION);
        bb.putShort((short) 1); // Not sure what the flag value is for...
        bb.putShort((short) unotif.length);
        bb.putShort((short) utitle.length);
        bb.putShort((short) umessage.length);
        bb.putShort((short) uappName.length);
        bb.put(unotif);
        bb.put(utitle);
        bb.put(umessage);
        bb.put(uappName);
        bb.put(md5(Arrays.copyOf(bb.array(), len - 16), password));

        return bb;
    }

    private void sendPacket(byte[] bytes) {
        try {
            DatagramSocket sct = new DatagramSocket();
            sct.connect(InetAddress.getByName(host), port);
            DatagramPacket pkt = new DatagramPacket(bytes, bytes.length);
            sct.send(pkt);
            sct.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    private void doRegistration(String appName, String password) {
        sendPacket(registrationPacket(appName, password).array());
    }

    public static GrowlNetwork register(String appName, String password) {
        return register(appName, password, DEFAULT_HOST, DEFAULT_PORT);
    }

    public static GrowlNetwork register(String appName, String password, String host) {
        return register(appName, password, host, DEFAULT_PORT);
    }

    public static GrowlNetwork register(String appName, String password, String host, int port) {
        GrowlNetwork g = new GrowlNetwork(host, port);
        g.doRegistration(appName, password);
        return g;
    }

    /**
     * test the grow
     *
     * @param args
     */
    public static void main(String[] args) {
        GrowlNetwork.notify("The title", "This is the notification message...");
    }
}