#include "ods.h"
 #include "dyndns.h"
 #include "dyns.h"
+#include "easydns.h"
 
 #include <time.h>
 #include <iostream>
  */
 Service::Ptr Config::create_service(const string &protocol,const string &hostname, const string &login, const string &password, const int update_interval, const int max_updates_within_interval, const int dns_cache_ttl)
 {
+    // Test for valid hostname. Must contain 3 parts minimum.
+    list<string> fqhn_parts;
+    ba::split(fqhn_parts,hostname,boost::is_any_of("."));
+    if ( fqhn_parts.size() < 3 )
+    {
+        Log->print_invalid_hostname(protocol);
+        Service::Ptr service;
+        return service;
+    }
+
     if(protocol == "dhs")
     {
         Service::Ptr service_dhs(new DHS(protocol,hostname,login,password,Log,update_interval,max_updates_within_interval,dns_cache_ttl,Proxy,ProxyPort));
         Service::Ptr service_dyns(new DYNS(protocol,hostname,login,password,Log,update_interval,max_updates_within_interval,dns_cache_ttl,Proxy,ProxyPort));
         return service_dyns;
     }
+    else if(protocol == "easydns")
+    {
+        Service::Ptr service_easydns(new EASYDNS(protocol,hostname,login,password,Log,update_interval,max_updates_within_interval,dns_cache_ttl,Proxy,ProxyPort));
+        return service_easydns;
+    }
     else
     {
         Log->print_unknown_protocol(protocol);
 
--- /dev/null
+/** @file
+ * @brief EASYDNS Service class implementation. This class represents the EASYDNS service.
+ *
+ *
+ *
+ * @copyright Intra2net AG
+ * @license GPLv2
+*/
+
+#include "easydns.h"
+
+#include <time.h>
+#include <boost/foreach.hpp>
+
+using namespace std;
+
+
+/**
+ * Default Constructor, needed for object serialization.
+ */
+EASYDNS::EASYDNS()
+{
+}
+
+
+/**
+ * Constructor.
+ * @param _hostname The hostname to update
+ * @param _login The login name.
+ * @param _password The corresponding password.
+ */
+EASYDNS::EASYDNS(const string& _protocol, const string& _hostname, const string& _login, const string& _password, const Logger::Ptr& _logger, const int _update_interval, const int _max_updates_within_interval, const int _dns_cache_ttl, const string& _proxy, const int _proxy_port)
+{
+    if ( _update_interval == -1 )        // If _update_interval is default po::option_desc (not specified via config)
+        set_update_interval(10);              // use default protocol value
+    else
+        set_update_interval(_update_interval);
+
+    if ( _max_updates_within_interval == -1 )
+        set_max_updates_within_interval(1);
+    else
+        set_max_updates_within_interval(_max_updates_within_interval);
+
+    if ( _dns_cache_ttl == -1 )
+        set_dns_cache_ttl(1200);
+    else
+        set_dns_cache_ttl(_dns_cache_ttl);
+
+    set_protocol(_protocol);
+    set_hostname(_hostname);
+    set_login(_login);
+    set_password(_password);
+    set_logger(_logger);
+
+    // create http helper class
+    HTTPHelper::Ptr _http_help(new HTTPHelper(_logger,_proxy,_proxy_port,_login,_password));
+    HTTPHelp = _http_help;
+    _http_help.reset();
+
+    // extract domain part from hostname
+    list<string> host_domain_part = separate_domain_and_host_part(get_hostname());
+
+    string two_level_tld = get_two_level_tld(host_domain_part.back());
+    if ( !two_level_tld.empty() )
+        BaseUrl = assemble_base_url(get_hostname(),two_level_tld);
+    else
+        BaseUrl = assemble_base_url(get_hostname(),"");
+}
+
+
+/**
+ * Default destructor
+ */
+EASYDNS::~EASYDNS()
+{
+}
+
+
+/**
+ * Tries to extract the two_level domain part if there is one
+ * @param domain_part The complete domain part.
+ * @return Two_level_domain part if there is one or "" if not so.
+ */
+string EASYDNS::get_two_level_tld(const string& domain_part) const
+{
+    // TODO: There is a list with all two level TLD's, use it
+
+    // split the domain_part
+    list<string> domain_tokens = split(domain_part,".");
+
+    domain_tokens.pop_front();
+
+    if ( domain_tokens.size() > 1 )
+    {
+        string two_level_tld = domain_tokens.front();
+        domain_tokens.pop_front();
+
+        BOOST_FOREACH(string domain_part, domain_tokens)
+        {
+            two_level_tld.append(".");
+            two_level_tld.append(domain_part);
+        }
+
+        return two_level_tld;
+    }
+    else
+    {
+        return "";
+    }
+}
+
+
+/**
+ * Assemble the easydns update url from the given hostname and domain part
+ * @param hostname The hostname to update IP for.
+ * @param domain_part The domain_part in which the hostname is located.
+ * @return The assembled update url without IP.
+ */
+string EASYDNS::assemble_base_url(const string& hostname, const string& two_level_tld) const
+{
+    string base_url;
+    if ( !two_level_tld.empty() )
+    {
+        base_url = "https://members.easydns.com";
+        base_url.append("/dyn/dyndns.php?hostname=");
+        base_url.append(hostname);
+        base_url.append("&tld=");
+        base_url.append(two_level_tld);
+        base_url.append("&myip=");
+    }
+    else
+    {
+        base_url = "https://members.easydns.com";
+        base_url.append("/dyn/dyndns.php?hostname=");
+        base_url.append(hostname);
+        base_url.append("&myip=");
+    }
+    return base_url;
+}
+
+
+/**
+ * Separates the hostname from the domain part.
+ * @param fqdn Hostname with domain part.
+ * @return A list with 2 elements (first element is the hostname, second element the domain part), or a list with 1 element if the domain part couldn't be determined.
+ */
+list<string> EASYDNS::separate_domain_and_host_part(const string& fqdn) const
+{
+    list<string> splitted = split(fqdn,".");
+
+    if ( splitted.size() > 1 )
+    {
+        string host = splitted.front();
+        splitted.pop_front();
+
+        string domain = splitted.front();
+        splitted.pop_front();
+
+        BOOST_FOREACH(string domain_part, splitted)
+        {
+            domain.append(".");
+            domain.append(domain_part);
+        }
+
+        splitted.clear();
+        splitted.push_back(host);
+        splitted.push_back(domain);
+    }
+
+    return splitted;
+}
+
+
+/**
+ * Splitts a string into single tokens useing given delimiters
+ * @param str String to split
+ * @param delimiters Deliminters to use
+ * @return A list with the single tokens
+ */
+list<string> EASYDNS::split(const string& str,const string& delimiters) const
+{
+    list<string> tokens;
+    // Skip delimiters at beginning.
+    string::size_type lastPos = str.find_first_not_of(delimiters, 0);
+    // Find first "non-delimiter".
+    string::size_type pos = str.find_first_of(delimiters, lastPos);
+
+    while (string::npos != pos || string::npos != lastPos)
+    {
+        // Found a token, add it to the list.
+        tokens.push_back(str.substr(lastPos, pos - lastPos));
+        // Skip delimiters.  Note the "not_of"
+        lastPos = str.find_first_not_of(delimiters, pos);
+        // Find next "non-delimiter"
+        pos = str.find_first_of(delimiters, lastPos);
+    }
+    return tokens;
+}
+
+
+/**
+ * Performs the Service update.
+ * @param ip IP Address to set.
+ * @return 0 if all is fine, -1 otherwise.
+ */
+int EASYDNS::perform_update(const std::string& ip)
+{
+    int ret_val = -1;
+
+    string url = BaseUrl;
+    url.append(ip);
+
+    long output = HTTPHelp->http_get(url);
+
+    get_logger()->print_http_status_code(url,output);
+
+    // HTTP operation completed successful, now we have to parse the data received by curl cause http status code is not significant for dyndns update errors
+    if ( output == 200 )
+    {
+        string curl_data = HTTPHelp->get_curl_data();
+
+        if ( curl_data == "NOERROR" )
+            ret_val = 0;
+        else if ( curl_data == "NOACCESS" )
+            get_logger()->print_http_not_authorized(url,get_login(),get_password());
+        else
+            get_logger()->print_update_failure(url, curl_data);
+    }
+
+    return ret_val;
+}
+
+
+/**
+ * Serialize function needed by boost/serialization to define which members should be stored as the object state.
+ * @param ar Archive
+ * @param version Version
+ */
+template<class Archive>
+void EASYDNS::serialize(Archive & ar, const unsigned int version)
+{
+    ar & boost::serialization::base_object<Service>(*this);
+}