summaryrefslogtreecommitdiff
path: root/webauth.c
blob: 30f78af21aa60b3b5f05a62de25049c543f14230 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/*
 * webauth.c
 *
 * See the README file for copyright information and how to reach the author.
 *
 */

// https://gnunet.org/svn/libmicrohttpd/doc/examples/tlsauthentication.c

#include "httpd.h"

//***************************************************************************
// String 2 Base 64
//***************************************************************************

char* string2base64(const char* message)
{
   const char *lookup = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
   unsigned long l;
   uint i;
   char* tmp;
   size_t length = strlen(message);

   tmp = (char*)malloc(length * 2);

   if (!tmp)
      return tmp;

   tmp[0] = 0;

   for (i = 0; i < length; i += 3)
   {
      l = (((unsigned long) message[i]) << 16)
         | (((i + 1) < length) ? (((unsigned long) message[i + 1]) << 8) : 0)
         | (((i + 2) < length) ? ((unsigned long) message[i + 2]) : 0);

      strncat (tmp, &lookup[(l >> 18) & 0x3F], 1);
      strncat (tmp, &lookup[(l >> 12) & 0x3F], 1);

      if (i + 1 < length)
         strncat(tmp, &lookup[(l >> 6) & 0x3F], 1);
      if (i + 2 < length)
         strncat(tmp, &lookup[l & 0x3F], 1);
   }

   if (length % 3)
      sprintf(tmp+strlen(tmp), "%.*s", (int)(3 - length % 3), "===");
      // strncat(tmp, "===", 3 - length % 3);

   return tmp;
}

//***************************************************************************
// Ask For Authentication
//***************************************************************************

int cEpgHttpd::askForAuthentication(struct MHD_Connection* connection, const char* realm)
{
   int status;
   struct MHD_Response* response;
   char* headervalue = 0;

   response = MHD_create_response_from_buffer(0, 0, MHD_RESPMEM_PERSISTENT);

   if (!response)
      return MHD_NO;

   asprintf(&headervalue, "Basic realm=\"%s\"", realm);
   status = MHD_add_response_header(response, "WWW-Authenticate", headervalue);
   free(headervalue);

   if (!status)
   {
      MHD_destroy_response(response);
      return MHD_NO;
   }

   status = MHD_queue_response(connection, MHD_HTTP_UNAUTHORIZED, response);

   MHD_destroy_response(response);

   return status;
}

//***************************************************************************
// Is Authenticated
//***************************************************************************

int cEpgHttpd::isAuthenticated(struct MHD_Connection* connection,
                               const char* username, const char* password)
{
   const char *headervalue;
   char *expected64, *expected;
   const char *strbase = "Basic ";
   int authenticated;

   headervalue = MHD_lookup_connection_value(connection, MHD_HEADER_KIND, "Authorization");

   if (!headervalue)
      return 0;

   if (strncmp(headervalue, strbase, strlen(strbase)) != 0)
      return 0;

   expected = (char*)malloc(strlen(username) + 1 + strlen(password) + 1);

   if (!expected)
      return 0;

   strcpy(expected, username);
   strcat(expected, ":");
   strcat(expected, password);

   expected64 = string2base64(expected);
   free(expected);

   if (!expected64)
      return 0;

   authenticated = (strcmp(headervalue + strlen(strbase), expected64) == 0);
   free(expected64);

   return authenticated;
}