/*
 * aimini.c
 *
 * Copyright (C) 2009-11 - ipoque GmbH
 * Copyright (C) 2011-22 - ntop.org
 *
 * This file is part of nDPI, an open source deep packet inspection
 * library based on the OpenDPI and PACE technology by ipoque GmbH
 *
 * nDPI is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * nDPI 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 Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with nDPI.  If not, see <http://www.gnu.org/licenses/>.
 * 
 */


#include "ndpi_protocol_ids.h"

#define NDPI_CURRENT_PROTO NDPI_PROTOCOL_AIMINI

#include "ndpi_api.h"


static void ndpi_int_aimini_add_connection(struct ndpi_detection_module_struct *ndpi_struct, struct ndpi_flow_struct *flow/* ,  */
					   /* ndpi_protocol_type_t protocol_type */)
{
  ndpi_set_detected_protocol(ndpi_struct, flow, NDPI_PROTOCOL_AIMINI, NDPI_PROTOCOL_HTTP, NDPI_CONFIDENCE_DPI);
}


void ndpi_search_aimini(struct ndpi_detection_module_struct *ndpi_struct, struct ndpi_flow_struct *flow)
{
  struct ndpi_packet_struct *packet = &ndpi_struct->packet;

  NDPI_LOG_DBG(ndpi_struct, "search aimini\n");

  if (packet->udp != NULL) {
    if (flow->l4.udp.aimini_stage == 0) {
      if (packet->payload_packet_len == 64 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x010b) {
	flow->l4.udp.aimini_stage = 1;
	NDPI_LOG_DBG2(ndpi_struct, "stage = 1\n");
	return;
      }
      if (packet->payload_packet_len == 136
	  && (ntohs(get_u_int16_t(packet->payload, 0)) == 0x01c9 || ntohs(get_u_int16_t(packet->payload, 0)) == 0x0165)) {
	flow->l4.udp.aimini_stage = 4;
	NDPI_LOG_DBG2(ndpi_struct, "stage = 4\n");
	return;
      }
      if (packet->payload_packet_len == 88 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0101) {
	flow->l4.udp.aimini_stage = 7;
	NDPI_LOG_DBG2(ndpi_struct, "stage = 7\n");
	return;
      }
      if (packet->payload_packet_len == 104 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0102) {
	flow->l4.udp.aimini_stage = 10;
	NDPI_LOG_DBG2(ndpi_struct, "stage = 10\n");
	return;
      }
      if (packet->payload_packet_len == 32 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x01ca) {
	flow->l4.udp.aimini_stage = 13;
	NDPI_LOG_DBG2(ndpi_struct, "stage = 13\n");
	return;
      }
      if (packet->payload_packet_len == 16 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x010c) {
	flow->l4.udp.aimini_stage = 16;
	NDPI_LOG_DBG2(ndpi_struct, "stage = 16\n");
	return;
      }
    }
    /* first packet chronology: (len, value): (64, 0x010b), (>100, 0x0115), (16, 0x010c || 64, 0x010b || 88, 0x0115),
     * (16, 0x010c || 64, 0x010b || >100, 0x0115)
     */
    if (flow->l4.udp.aimini_stage == 1 && packet->payload_packet_len > 100
	&& ntohs(get_u_int16_t(packet->payload, 0)) == 0x0115) {
      flow->l4.udp.aimini_stage = 2;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 2\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 2 &&
	((packet->payload_packet_len == 16 && get_u_int16_t(packet->payload, 0) == htons(0x010c)) ||
	 (packet->payload_packet_len == 64 && get_u_int16_t(packet->payload, 0) == htons(0x010b)) ||
	 (packet->payload_packet_len == 88 && get_u_int16_t(packet->payload, 0) == ntohs(0x0115)))) {
      flow->l4.udp.aimini_stage = 3;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 3\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 3
	&& ((packet->payload_packet_len == 16 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x010c)
	    || (packet->payload_packet_len == 64 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x010b)
	    || (packet->payload_packet_len > 100 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0115))) {
      NDPI_LOG_INFO(ndpi_struct, "found aimini (64, 0x010b), (>300, 0x0115), "
		    "(16, 0x010c || 64, 0x010b), (16, 0x010c || 64, 0x010b || >100, 0x0115).\n");
      ndpi_int_aimini_add_connection(ndpi_struct, flow);
      return;
    }

    /* second packet chronology: (len, value): (136, 0x01c9), (136, 0x01c9),(136, 0x01c9),(136, 0x01c9 || 32, 0x01ca) */

    if (flow->l4.udp.aimini_stage == 4 && packet->payload_packet_len == 136
	&& (ntohs(get_u_int16_t(packet->payload, 0)) == 0x01c9 || ntohs(get_u_int16_t(packet->payload, 0)) == 0x0165)) {
      flow->l4.udp.aimini_stage = 5;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 5\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 5 && (packet->payload_packet_len == 136
					   && (ntohs(get_u_int16_t(packet->payload, 0)) == 0x01c9
					       || ntohs(get_u_int16_t(packet->payload, 0)) == 0x0165))) {
      flow->l4.udp.aimini_stage = 6;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 6\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 6 && ((packet->payload_packet_len == 136
					    && ((ntohs(get_u_int16_t(packet->payload, 0)) == 0x0165)
						|| ntohs(get_u_int16_t(packet->payload, 0)) == 0x01c9))
					   || (packet->payload_packet_len == 32
					       && ntohs(get_u_int16_t(packet->payload, 0)) == 0x01ca))) {
      NDPI_LOG_INFO(ndpi_struct,
		    "found aimini (136, 0x01c9), (136, 0x01c9)," "(136, 0x01c9),(136, 0x01c9 || 32, 0x01ca).\n");
      ndpi_int_aimini_add_connection(ndpi_struct, flow);
      return;
    }

    /* third packet chronology: (len, value): (88, 0x0101), (88, 0x0101),(88, 0x0101),(88, 0x0101) */

    if (flow->l4.udp.aimini_stage == 7 && packet->payload_packet_len == 88
	&& ntohs(get_u_int16_t(packet->payload, 0)) == 0x0101) {
      flow->l4.udp.aimini_stage = 8;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 8\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 8
	&& (packet->payload_packet_len == 88 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0101)) {
      flow->l4.udp.aimini_stage = 9;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 9\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 9
	&& (packet->payload_packet_len == 88 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0101)) {
      NDPI_LOG_INFO(ndpi_struct,
		    "found aimini (88, 0x0101), (88, 0x0101)," "(88, 0x0101),(88, 0x0101).\n");
      ndpi_int_aimini_add_connection(ndpi_struct, flow);
      return;
    }

    /* fourth packet chronology: (len, value): (104, 0x0102), (104, 0x0102), (104, 0x0102), (104, 0x0102) */

    if (flow->l4.udp.aimini_stage == 10 && packet->payload_packet_len == 104
	&& ntohs(get_u_int16_t(packet->payload, 0)) == 0x0102) {
      flow->l4.udp.aimini_stage = 11;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 11\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 11
	&& (packet->payload_packet_len == 104 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0102)) {
      flow->l4.udp.aimini_stage = 12;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 12\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 12
	&& ((packet->payload_packet_len == 104 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0102)
	    || (packet->payload_packet_len == 32 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x01ca))) {
      NDPI_LOG_INFO(ndpi_struct,
		    "found aimini (104, 0x0102), (104, 0x0102), " "(104, 0x0102), (104, 0x0102).\n");
      ndpi_int_aimini_add_connection(ndpi_struct, flow);
      return;
    }

    /* fifth packet chronology (len, value): (32,0x01ca), (32,0x01ca), (32,0x01ca), ((136, 0x0166) || (32,0x01ca)) */

    if (flow->l4.udp.aimini_stage == 13 && packet->payload_packet_len == 32
	&& ntohs(get_u_int16_t(packet->payload, 0)) == 0x01ca) {
      flow->l4.udp.aimini_stage = 14;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 14\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 14
	&& ((packet->payload_packet_len == 32 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x01ca)
	    || (packet->payload_packet_len == 136 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0166))) {
      flow->l4.udp.aimini_stage = 15;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 15\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 15
	&& ((packet->payload_packet_len == 136 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x0166)
	    || (packet->payload_packet_len == 32 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x01ca))) {
      NDPI_LOG_INFO(ndpi_struct,
		    "found aimini (32,0x01ca), (32,0x01ca), (32,0x01ca), ((136, 0x0166)||(32,0x01ca)).\n");
      ndpi_int_aimini_add_connection(ndpi_struct, flow);
      return;
    }

    /* sixth packet chronology (len, value): (16, 0x010c), (16, 0x010c), (16, 0x010c), (16, 0x010c) */

    if (flow->l4.udp.aimini_stage == 16 && packet->payload_packet_len == 16
	&& ntohs(get_u_int16_t(packet->payload, 0)) == 0x010c) {
      flow->l4.udp.aimini_stage = 17;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 17\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 17
	&& (packet->payload_packet_len == 16 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x010c)) {
      flow->l4.udp.aimini_stage = 18;
      NDPI_LOG_DBG2(ndpi_struct, "stage = 18\n");
      return;
    }
    if (flow->l4.udp.aimini_stage == 18
	&& (packet->payload_packet_len == 16 && ntohs(get_u_int16_t(packet->payload, 0)) == 0x010c)) {
      NDPI_LOG_INFO(ndpi_struct,
		    "found aimini (16, 0x010c), (16, 0x010c), (16, 0x010c), (16, 0x010c).\n");
      ndpi_int_aimini_add_connection(ndpi_struct, flow);
      return;
    }
  }

  if (flow->detected_protocol_stack[0] == NDPI_PROTOCOL_HTTP) {
    if (flow->http.method == NDPI_HTTP_METHOD_GET)
      {
	if ((LINE_STARTS(packet->http_url_name, "/download/") == 1 ||
	     LINE_STARTS(packet->http_url_name, "/player/") == 1 ||
	     LINE_STARTS(packet->http_url_name, "/webcounter/") == 1 ||
	     LINE_STARTS(packet->http_url_name, "/play/") == 1 ||
	     LINE_STARTS(packet->http_url_name, "/search/") == 1 ||
	     LINE_STARTS(packet->http_url_name, "/member/") == 1) &&
	    (LINE_ENDS(packet->host_line, ".aimini.net") == 1 ||
	     LINE_ENDS(packet->host_line, ".aimini.com") == 1))
	  {
	    NDPI_LOG_INFO(ndpi_struct, "found AIMINI HTTP traffic\n");
	    ndpi_int_aimini_add_connection(ndpi_struct, flow);
	    return;
	  }
      } else if (flow->http.method == NDPI_HTTP_METHOD_POST)
      {
	if ((LINE_STARTS(packet->http_url_name, "/upload/") == 1 ||
	     LINE_STARTS(packet->http_url_name, "/member/") == 1) &&
	    (LINE_ENDS(packet->host_line, ".aimini.net") == 1 ||
	     LINE_ENDS(packet->host_line, ".aimini.com") == 1))
	  {
	    NDPI_LOG_INFO(ndpi_struct, "found AIMINI HTTP traffic\n");
	    ndpi_int_aimini_add_connection(ndpi_struct, flow);
	    return;
	  }
      }
  }

  NDPI_EXCLUDE_PROTO(ndpi_struct, flow);

}


void init_aimini_dissector(struct ndpi_detection_module_struct *ndpi_struct, u_int32_t *id, NDPI_PROTOCOL_BITMASK *detection_bitmask)
{
  ndpi_set_bitmask_protocol_detection("Aimini", ndpi_struct, detection_bitmask, *id,
				      NDPI_PROTOCOL_AIMINI,
				      ndpi_search_aimini,
				      NDPI_SELECTION_BITMASK_PROTOCOL_V4_V6_TCP_OR_UDP_WITH_PAYLOAD_WITHOUT_RETRANSMISSION,
				      SAVE_DETECTION_BITMASK_AS_UNKNOWN,
				      ADD_TO_DETECTION_BITMASK);

  *id += 1;
}