summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--configure.ac47
-rw-r--r--include/opkele/exception.h9
-rw-r--r--lib/Makefile.am4
-rw-r--r--lib/consumer.cc124
-rw-r--r--libopkele.pc.in2
5 files changed, 80 insertions, 106 deletions
diff --git a/configure.ac b/configure.ac
index 00c6bc4..53e22ba 100644
--- a/configure.ac
+++ b/configure.ac
@@ -1,132 +1,91 @@
1AC_INIT([libopkele], [0.3], [libopkele-bugs@klever.net]) 1AC_INIT([libopkele], [0.3], [libopkele-bugs@klever.net])
2AC_CONFIG_SRCDIR([include/opkele/opkele-config.h]) 2AC_CONFIG_SRCDIR([include/opkele/opkele-config.h])
3AC_CONFIG_HEADERS([config.h include/opkele/acconfig.h]) 3AC_CONFIG_HEADERS([config.h include/opkele/acconfig.h])
4AM_INIT_AUTOMAKE([dist-bzip2]) 4AM_INIT_AUTOMAKE([dist-bzip2])
5 5
6AC_PROG_INSTALL 6AC_PROG_INSTALL
7AC_PROG_CXX 7AC_PROG_CXX
8AC_PROG_CC 8AC_PROG_CC
9AC_PROG_LIBTOOL 9AC_PROG_LIBTOOL
10 10
11AC_HEADER_STDC 11AC_HEADER_STDC
12 12
13AC_PATH_PROG([XSLTPROC],[xsltproc],[true]) 13AC_PATH_PROG([XSLTPROC],[xsltproc],[true])
14AC_WITH_PKGCONFIG 14AC_WITH_PKGCONFIG
15 15
16PKG_CHECK_MODULES([OPENSSL],[openssl],,[ 16PKG_CHECK_MODULES([OPENSSL],[openssl],,[
17 AC_MSG_ERROR([no openssl library found. get one from http://www.openssl.org/]) 17 AC_MSG_ERROR([no openssl library found. get one from http://www.openssl.org/])
18]) 18])
19 19
20WANT_KONFORKA="yes" 20WANT_KONFORKA="yes"
21AC_ARG_ENABLE([konforka], 21AC_ARG_ENABLE([konforka],
22 AC_HELP_STRING([--disable-konforka],[do not use konforka library (default: use if found)]), 22 AC_HELP_STRING([--disable-konforka],[do not use konforka library (default: use if found)]),
23 [ 23 [
24 test "${enableval}" = "no" && WANT_KONFORKA="no" 24 test "${enableval}" = "no" && WANT_KONFORKA="no"
25 ] 25 ]
26) 26)
27if test "${WANT_KONFORKA}" = "yes" ; then 27if test "${WANT_KONFORKA}" = "yes" ; then
28 PKG_CHECK_MODULES([KONFORKA],[konforka],[ 28 PKG_CHECK_MODULES([KONFORKA],[konforka],[
29 AC_SUBST([KONFORKA_CFLAGS]) 29 AC_SUBST([KONFORKA_CFLAGS])
30 AC_SUBST([KONFORKA_LIBS]) 30 AC_SUBST([KONFORKA_LIBS])
31 AC_DEFINE([HAVE_KONFORKA],,[defined in presence of konforka library]) 31 AC_DEFINE([HAVE_KONFORKA],,[defined in presence of konforka library])
32 AC_DEFINE([OPKELE_HAVE_KONFORKA],,[defined in presence of konforka library]) 32 AC_DEFINE([OPKELE_HAVE_KONFORKA],,[defined in presence of konforka library])
33 AC_SUBST([KONFORKA_KONFORKA],[konforka]) 33 AC_SUBST([KONFORKA_KONFORKA],[konforka])
34 ],[true]) 34 ],[true])
35fi 35fi
36 36
37WANT_DOXYGEN="yes" 37WANT_DOXYGEN="yes"
38AC_ARG_ENABLE([doxygen], 38AC_ARG_ENABLE([doxygen],
39 AC_HELP_STRING([--disable-doxygen],[do not generate documentation]), 39 AC_HELP_STRING([--disable-doxygen],[do not generate documentation]),
40 [ 40 [
41 test "${enableval}" = "no" && WANT_DOXYGEN="no" 41 test "${enableval}" = "no" && WANT_DOXYGEN="no"
42 ] 42 ]
43) 43)
44if test "${WANT_DOXYGEN}" = "yes" ; then 44if test "${WANT_DOXYGEN}" = "yes" ; then
45 AC_WITH_DOXYGEN 45 AC_WITH_DOXYGEN
46 AC_WITH_DOT 46 AC_WITH_DOT
47else 47else
48 AM_CONDITIONAL([HAVE_DOXYGEN],[false]) 48 AM_CONDITIONAL([HAVE_DOXYGEN],[false])
49 AM_CONDITIONAL([HAVE_DOT],[false]) 49 AM_CONDITIONAL([HAVE_DOT],[false])
50fi 50fi
51 51
52LIBCURL_CHECK_CONFIG(,,,[ 52LIBCURL_CHECK_CONFIG(,,,[
53 AC_MSG_ERROR([no required libcurl library. get one from http://curl.haxx.se/]) 53 AC_MSG_ERROR([no required libcurl library. get one from http://curl.haxx.se/])
54]) 54])
55 55
56want_pcre_impl="" 56PKG_CHECK_MODULES([PCRE],[libpcre],,[
57AC_ARG_WITH([pcre-bindings], 57 AC_MSG_ERROR([no libpcre found, go get it at http://www.pcre.org/])
58 AC_HELP_STRING([--with-pcre-bindings=(pcrepp|none|libpcrecpp)],[Specify which pcre c++ bindings to use. 'pcrepp' stands for quite sensible library, found at http://www.daemon.de/PCRE/, 'libcrecpp' makes use of crappy bindings by google and 'none' disables internal implementation of OP discovery]), 58])
59 [
60 case "$withval" in
61 pcrepp) want_pcre_impl="pcrepp" ;;
62 libpcrecpp) want_pcre_impl="libpcrecpp" ;;
63 none) want_pcre_impl="none";;
64 *) AC_MSG_ERROR([I'm not sure I understand what do you want for a pcre c++ bindings]) ;;
65 esac
66 ]
67)
68
69found_pcre_impl=""
70
71if test -z "$want_pcre_impl" -o "$want_pcre_impl" = "pcrepp" ; then
72 AC_WITH_PCRE([
73 AC_WITH_PCREPP([
74 found_pcre_impl=pcrepp
75 ])
76 ],[
77 AC_MSG_ERROR([no pcre library found. get one at http://www.pcre.org/])
78 ]
79 )
80 test "$want_pcre_impl,$found_pcre_impl" = "pcrepp," && AC_MSG_ERROR([no pcre++ library found. get one at http://www.daemon.de/PCRE])
81fi
82
83if test "$found_pcre_impl,$want_pcre_impl" = "," -o "$want_pcre_impl" = "libpcrecpp" ; then
84 test -z "$want_pcre_impl" || AC_MSG_NOTICE([You want to use crappy libpcre c++ bindings])
85 PKG_CHECK_MODULES([LIBPCRECPP],[libpcrecpp],[
86 found_pcre_impl=libpcrecpp
87 CXXFLAGS="$CXXFLAGS $LIBPCRECPP_CFLAGS"
88 LIBS="$LIBS $LIBPCRECPP_LIBS"
89 ],[
90 test -z "$want_pcre_impl" || AC_MSG_ERROR([no libpcre c++ bindings found. why would you want it if you don't have it installed?])
91 ]
92 )
93fi
94test "$want_pcre_impl,$found_pcre_impl" = "," && AC_MSG_ERROR([no pcre c++ bindings found, use --with-pcre-bindings=none to disable code that makes use of it])
95
96case "$found_pcre_impl" in
97 pcrepp) AC_DEFINE([USE_PCREPP],,[defined if pcre++ is to be used]) ;;
98 libpcrecpp) AC_DEFINE([USE_LIBPCRECPP],,[defined if crappy google bindings are to be used]) ;;
99esac
100 59
101curl_ssl_verify_host="true" 60curl_ssl_verify_host="true"
102AC_ARG_ENABLE([ssl-verify-host], 61AC_ARG_ENABLE([ssl-verify-host],
103 AC_HELP_STRING([--disable-ssl-verify-host],[disable cURL cert/host relationships verification]), 62 AC_HELP_STRING([--disable-ssl-verify-host],[disable cURL cert/host relationships verification]),
104 [ test "${enableval}" = "no" && curl_ssl_verify_host="false" ] 63 [ test "${enableval}" = "no" && curl_ssl_verify_host="false" ]
105) 64)
106${curl_ssl_verify_host} || AC_DEFINE([DISABLE_CURL_SSL_VERIFYHOST],,[defined if cURL is not to verify cert/host]) 65${curl_ssl_verify_host} || AC_DEFINE([DISABLE_CURL_SSL_VERIFYHOST],,[defined if cURL is not to verify cert/host])
107 66
108curl_ssl_verify_peer="true" 67curl_ssl_verify_peer="true"
109AC_ARG_ENABLE([ssl-verify-peer], 68AC_ARG_ENABLE([ssl-verify-peer],
110 AC_HELP_STRING([--disable-ssl-verify-peer],[disable cURL cert validity verification]), 69 AC_HELP_STRING([--disable-ssl-verify-peer],[disable cURL cert validity verification]),
111 [ test "${enableval}" = "no" && curl_ssl_verify_peer="false" ] 70 [ test "${enableval}" = "no" && curl_ssl_verify_peer="false" ]
112) 71)
113${curl_ssl_verify_peer} || AC_DEFINE([DISABLE_CURL_SSL_VERIFYPEER],,[defined if cURL is not to verify cert validity]) 72${curl_ssl_verify_peer} || AC_DEFINE([DISABLE_CURL_SSL_VERIFYPEER],,[defined if cURL is not to verify cert validity])
114 73
115postels_law=true 74postels_law=true
116AC_ARG_ENABLE([postels-law], 75AC_ARG_ENABLE([postels-law],
117 AC_HELP_STRING([--disable-postels-law],[Be strict, do not adhere to Postel's Law ("be conservative in what you do, be liberal in what you accept from others", RFC 793)]), 76 AC_HELP_STRING([--disable-postels-law],[Be strict, do not adhere to Postel's Law ("be conservative in what you do, be liberal in what you accept from others", RFC 793)]),
118 [ test "${enableval}" = "no" && postels_law=false ] 77 [ test "${enableval}" = "no" && postels_law=false ]
119) 78)
120$postels_law && AC_DEFINE([POSTELS_LAW],,[defined if we want to adhere to Postel's Law]) 79$postels_law && AC_DEFINE([POSTELS_LAW],,[defined if we want to adhere to Postel's Law])
121 80
122AC_DEFINE_UNQUOTED([OPKELE_SRC_DIR],["$PWD"],[source directory]) 81AC_DEFINE_UNQUOTED([OPKELE_SRC_DIR],["$PWD"],[source directory])
123 82
124AC_CONFIG_FILES([ 83AC_CONFIG_FILES([
125 Makefile 84 Makefile
126 libopkele.pc 85 libopkele.pc
127 Doxyfile 86 Doxyfile
128 include/Makefile 87 include/Makefile
129 lib/Makefile 88 lib/Makefile
130 test/Makefile 89 test/Makefile
131]) 90])
132AC_OUTPUT 91AC_OUTPUT
diff --git a/include/opkele/exception.h b/include/opkele/exception.h
index 0150e6b..753a818 100644
--- a/include/opkele/exception.h
+++ b/include/opkele/exception.h
@@ -23,195 +23,204 @@
23 * the exception codepoint specification 23 * the exception codepoint specification
24 */ 24 */
25# define OPKELE_CP_ CODEPOINT, 25# define OPKELE_CP_ CODEPOINT,
26/** 26/**
27 * the simple rethrow of konforka-based exception 27 * the simple rethrow of konforka-based exception
28 */ 28 */
29# define OPKELE_RETHROW catch(konforka::exception& e) { e.see(CODEPOINT); throw } 29# define OPKELE_RETHROW catch(konforka::exception& e) { e.see(CODEPOINT); throw }
30#else /* OPKELE_HAVE_KONFORKA */ 30#else /* OPKELE_HAVE_KONFORKA */
31# include <exception> 31# include <exception>
32# include <string> 32# include <string>
33/** 33/**
34 * the exception parameter declaration 34 * the exception parameter declaration
35 */ 35 */
36# define OPKELE_E_PARS const string& w 36# define OPKELE_E_PARS const string& w
37/** 37/**
38 * the dummy prefix for exception parameters list to prepend in the absence of 38 * the dummy prefix for exception parameters list to prepend in the absence of
39 * konforka library 39 * konforka library
40 */ 40 */
41# define OPKELE_E_CONS_ 41# define OPKELE_E_CONS_
42/** 42/**
43 * the dummy placeholder for konforka exception codepoint specification 43 * the dummy placeholder for konforka exception codepoint specification
44 */ 44 */
45# define OPKELE_CP_ 45# define OPKELE_CP_
46/** 46/**
47 * the dummy define for the konforka-based rethrow of exception 47 * the dummy define for the konforka-based rethrow of exception
48 */ 48 */
49# define OPKELE_RETHROW 49# define OPKELE_RETHROW
50#endif /* OPKELE_HAVE_KONFORKA */ 50#endif /* OPKELE_HAVE_KONFORKA */
51/** 51/**
52 * the exception parameters list to pass to constructor 52 * the exception parameters list to pass to constructor
53 */ 53 */
54# define OPKELE_E_CONS OPKELE_E_CONS_ w 54# define OPKELE_E_CONS OPKELE_E_CONS_ w
55 55
56namespace opkele { 56namespace opkele {
57 using std::string; 57 using std::string;
58 58
59 /** 59 /**
60 * the base opkele exception class 60 * the base opkele exception class
61 */ 61 */
62 class exception : public 62 class exception : public
63# ifdef OPKELE_HAVE_KONFORKA 63# ifdef OPKELE_HAVE_KONFORKA
64 konforka::exception 64 konforka::exception
65# else 65# else
66 std::exception 66 std::exception
67# endif 67# endif
68 { 68 {
69 public: 69 public:
70# ifdef OPKELE_HAVE_KONFORKA 70# ifdef OPKELE_HAVE_KONFORKA
71 explicit 71 explicit
72 exception(const string& fi,const string& fu,int l,const string& w) 72 exception(const string& fi,const string& fu,int l,const string& w)
73 : konforka::exception(fi,fu,l,w) { } 73 : konforka::exception(fi,fu,l,w) { }
74# else /* OPKELE_HAVE_KONFORKA */ 74# else /* OPKELE_HAVE_KONFORKA */
75 string _what; 75 string _what;
76 explicit 76 explicit
77 exception(const string& w) 77 exception(const string& w)
78 : _what(w) { } 78 : _what(w) { }
79 virtual ~exception() throw(); 79 virtual ~exception() throw();
80 virtual const char * what() const throw(); 80 virtual const char * what() const throw();
81# endif /* OPKELE_HAVE_KONFORKA */ 81# endif /* OPKELE_HAVE_KONFORKA */
82 }; 82 };
83 83
84 /** 84 /**
85 * thrown in case of failed conversion 85 * thrown in case of failed conversion
86 */ 86 */
87 class failed_conversion : public exception { 87 class failed_conversion : public exception {
88 public: 88 public:
89 failed_conversion(OPKELE_E_PARS) 89 failed_conversion(OPKELE_E_PARS)
90 : exception(OPKELE_E_CONS) { } 90 : exception(OPKELE_E_CONS) { }
91 }; 91 };
92 /** 92 /**
93 * thrown in case of failed lookup (either parameter or persistent store) 93 * thrown in case of failed lookup (either parameter or persistent store)
94 */ 94 */
95 class failed_lookup : public exception { 95 class failed_lookup : public exception {
96 public: 96 public:
97 failed_lookup(OPKELE_E_PARS) 97 failed_lookup(OPKELE_E_PARS)
98 : exception(OPKELE_E_CONS) { } 98 : exception(OPKELE_E_CONS) { }
99 }; 99 };
100 /** 100 /**
101 * thrown in case of bad input (either local or network) 101 * thrown in case of bad input (either local or network)
102 */ 102 */
103 class bad_input : public exception { 103 class bad_input : public exception {
104 public: 104 public:
105 bad_input(OPKELE_E_PARS) 105 bad_input(OPKELE_E_PARS)
106 : exception(OPKELE_E_CONS) { } 106 : exception(OPKELE_E_CONS) { }
107 }; 107 };
108 108
109 /** 109 /**
110 * thrown on failed assertion 110 * thrown on failed assertion
111 */ 111 */
112 class failed_assertion : public exception { 112 class failed_assertion : public exception {
113 public: 113 public:
114 failed_assertion(OPKELE_E_PARS) 114 failed_assertion(OPKELE_E_PARS)
115 : exception(OPKELE_E_CONS) { } 115 : exception(OPKELE_E_CONS) { }
116 }; 116 };
117 117
118 /** 118 /**
119 * thrown if the handle being retrieved is invalid 119 * thrown if the handle being retrieved is invalid
120 */ 120 */
121 class invalid_handle : public exception { 121 class invalid_handle : public exception {
122 public: 122 public:
123 invalid_handle(OPKELE_E_PARS) 123 invalid_handle(OPKELE_E_PARS)
124 : exception(OPKELE_E_CONS) { } 124 : exception(OPKELE_E_CONS) { }
125 }; 125 };
126 /** 126 /**
127 * thrown if the handle passed to check_authentication request is not 127 * thrown if the handle passed to check_authentication request is not
128 * stateless 128 * stateless
129 */ 129 */
130 class stateful_handle : public exception { 130 class stateful_handle : public exception {
131 public: 131 public:
132 stateful_handle(OPKELE_E_PARS) 132 stateful_handle(OPKELE_E_PARS)
133 : exception(OPKELE_E_CONS) { } 133 : exception(OPKELE_E_CONS) { }
134 }; 134 };
135 135
136 /** 136 /**
137 * thrown if check_authentication request fails 137 * thrown if check_authentication request fails
138 */ 138 */
139 class failed_check_authentication : public exception { 139 class failed_check_authentication : public exception {
140 public: 140 public:
141 failed_check_authentication(OPKELE_E_PARS) 141 failed_check_authentication(OPKELE_E_PARS)
142 : exception(OPKELE_E_CONS) { } 142 : exception(OPKELE_E_CONS) { }
143 }; 143 };
144 144
145 /** 145 /**
146 * thrown if the id_res request result is negative 146 * thrown if the id_res request result is negative
147 */ 147 */
148 class id_res_failed : public exception { 148 class id_res_failed : public exception {
149 public: 149 public:
150 id_res_failed(OPKELE_E_PARS) 150 id_res_failed(OPKELE_E_PARS)
151 : exception(OPKELE_E_CONS) { } 151 : exception(OPKELE_E_CONS) { }
152 }; 152 };
153 /** 153 /**
154 * thrown if the user_setup_url is provided with negative response 154 * thrown if the user_setup_url is provided with negative response
155 */ 155 */
156 class id_res_setup : public id_res_failed { 156 class id_res_setup : public id_res_failed {
157 public: 157 public:
158 string setup_url; 158 string setup_url;
159 id_res_setup(OPKELE_E_PARS,const string& su) 159 id_res_setup(OPKELE_E_PARS,const string& su)
160 : id_res_failed(OPKELE_E_CONS), setup_url(su) { } 160 : id_res_failed(OPKELE_E_CONS), setup_url(su) { }
161 ~id_res_setup() throw() { } 161 ~id_res_setup() throw() { }
162 }; 162 };
163 /** 163 /**
164 * thrown in case of signature mismatch 164 * thrown in case of signature mismatch
165 */ 165 */
166 class id_res_mismatch : public id_res_failed { 166 class id_res_mismatch : public id_res_failed {
167 public: 167 public:
168 id_res_mismatch(OPKELE_E_PARS) 168 id_res_mismatch(OPKELE_E_PARS)
169 : id_res_failed(OPKELE_E_CONS) { } 169 : id_res_failed(OPKELE_E_CONS) { }
170 }; 170 };
171 171
172 /** 172 /**
173 * openssl malfunction occured 173 * openssl malfunction occured
174 */ 174 */
175 class exception_openssl : public exception { 175 class exception_openssl : public exception {
176 public: 176 public:
177 unsigned long _error; 177 unsigned long _error;
178 string _ssl_string; 178 string _ssl_string;
179 exception_openssl(OPKELE_E_PARS); 179 exception_openssl(OPKELE_E_PARS);
180 ~exception_openssl() throw() { } 180 ~exception_openssl() throw() { }
181 }; 181 };
182 182
183 /** 183 /**
184 * network operation related error occured 184 * network operation related error occured
185 */ 185 */
186 class exception_network : public exception { 186 class exception_network : public exception {
187 public: 187 public:
188 exception_network(OPKELE_E_PARS) 188 exception_network(OPKELE_E_PARS)
189 : exception(OPKELE_E_CONS) { } 189 : exception(OPKELE_E_CONS) { }
190 }; 190 };
191 191
192 /** 192 /**
193 * network operation related error occured, specifically, related to 193 * network operation related error occured, specifically, related to
194 * libcurl 194 * libcurl
195 */ 195 */
196 class exception_curl : public exception_network { 196 class exception_curl : public exception_network {
197 public: 197 public:
198 CURLcode _error; 198 CURLcode _error;
199 string _curl_string; 199 string _curl_string;
200 exception_curl(OPKELE_E_PARS); 200 exception_curl(OPKELE_E_PARS);
201 exception_curl(OPKELE_E_PARS,CURLcode e); 201 exception_curl(OPKELE_E_PARS,CURLcode e);
202 ~exception_curl() throw() { } 202 ~exception_curl() throw() { }
203 }; 203 };
204 204
205 /** 205 /**
206 * not implemented (think pure virtual) member function executed, signfies 206 * not implemented (think pure virtual) member function executed, signfies
207 * programmer error 207 * programmer error
208 */ 208 */
209 class not_implemented : public exception { 209 class not_implemented : public exception {
210 public: 210 public:
211 not_implemented(OPKELE_E_PARS) 211 not_implemented(OPKELE_E_PARS)
212 : exception(OPKELE_E_CONS) { } 212 : exception(OPKELE_E_CONS) { }
213 }; 213 };
214 214
215 /**
216 * internal error, indicates internal libopkele problem
217 */
218 class internal_error : public exception {
219 public:
220 internal_error(OPKELE_E_PARS)
221 : exception(OPKELE_E_CONS) { }
222 };
223
215} 224}
216 225
217#endif /* __OPKELE_EXCEPTION_H */ 226#endif /* __OPKELE_EXCEPTION_H */
diff --git a/lib/Makefile.am b/lib/Makefile.am
index a68faf6..b008a52 100644
--- a/lib/Makefile.am
+++ b/lib/Makefile.am
@@ -1,27 +1,27 @@
1lib_LTLIBRARIES = libopkele.la 1lib_LTLIBRARIES = libopkele.la
2 2
3INCLUDES = \ 3INCLUDES = \
4 -I${top_srcdir}/include/ \ 4 -I${top_srcdir}/include/ \
5 ${KONFORKA_CFLAGS} \ 5 ${KONFORKA_CFLAGS} \
6 ${OPENSSL_CFLAGS} \ 6 ${OPENSSL_CFLAGS} \
7 ${LIBCURL_CPPFLAGS} \ 7 ${LIBCURL_CPPFLAGS} \
8 ${PCREPP_CFLAGS} 8 ${PCRE_CFLAGS}
9libopkele_la_LIBADD = \ 9libopkele_la_LIBADD = \
10 ${LIBCURL} \ 10 ${LIBCURL} \
11 ${PCREPP_LIBS} \ 11 ${PCRE_LIBS} \
12 ${OPENSSL_LIBS} \ 12 ${OPENSSL_LIBS} \
13 ${KONFORKA_LIBS} 13 ${KONFORKA_LIBS}
14 14
15libopkele_la_SOURCES = \ 15libopkele_la_SOURCES = \
16 params.cc \ 16 params.cc \
17 util.cc \ 17 util.cc \
18 server.cc \ 18 server.cc \
19 secret.cc \ 19 secret.cc \
20 data.cc \ 20 data.cc \
21 consumer.cc \ 21 consumer.cc \
22 exception.cc \ 22 exception.cc \
23 extension.cc \ 23 extension.cc \
24 sreg.cc \ 24 sreg.cc \
25 extension_chain.cc 25 extension_chain.cc
26libopkele_la_LDFLAGS = \ 26libopkele_la_LDFLAGS = \
27 -version-info 2:0:0 27 -version-info 2:0:0
diff --git a/lib/consumer.cc b/lib/consumer.cc
index ff5da91..df95b64 100644
--- a/lib/consumer.cc
+++ b/lib/consumer.cc
@@ -1,412 +1,418 @@
1#include <algorithm> 1#include <algorithm>
2#include <cassert> 2#include <cassert>
3#include <opkele/util.h> 3#include <opkele/util.h>
4#include <opkele/exception.h> 4#include <opkele/exception.h>
5#include <opkele/data.h> 5#include <opkele/data.h>
6#include <opkele/consumer.h> 6#include <opkele/consumer.h>
7#include <openssl/sha.h> 7#include <openssl/sha.h>
8#include <openssl/hmac.h> 8#include <openssl/hmac.h>
9#include <curl/curl.h> 9#include <curl/curl.h>
10 10
11#include <iostream> 11#include <iostream>
12 12
13#include "config.h" 13#include "config.h"
14 14
15#if defined(USE_LIBPCRECPP) 15#include <pcre.h>
16# include <pcrecpp.h>
17#elif defined(USE_PCREPP)
18# include <pcre++.h>
19#else
20 /* internal implementation won't be built */
21#endif
22 16
23namespace opkele { 17namespace opkele {
24 using namespace std; 18 using namespace std;
25 19
20 class pcre_matches_t {
21 public:
22 int *_ov;
23 int _s;
24
25 pcre_matches_t() : _ov(0), _s(0) { }
26 pcre_matches_t(int s) : _ov(0), _s(s) {
27 if(_s&1) ++_s;
28 _s += _s>>1;
29 _ov = new int[_s];
30 }
31 ~pcre_matches_t() throw() { if(_ov) delete[] _ov; }
32
33 int begin(int i) const { return _ov[i<<1]; }
34 int end(int i) const { return _ov[(i<<1)+1]; }
35 int length(int i) const { int t=i<<1; return _ov[t+1]-_ov[t]; }
36 };
37
38 class pcre_t {
39 public:
40 pcre *_p;
41
42 pcre_t() : _p(0) { }
43 pcre_t(pcre *p) : _p(p) { }
44 pcre_t(const char *re,int opts) : _p(0) {
45 static const char *errptr; static int erroffset;
46 _p = pcre_compile(re,opts,&errptr,&erroffset,NULL);
47 if(!_p)
48 throw internal_error(OPKELE_CP_ string("Failed to compile regexp: ")+errptr);
49 }
50 ~pcre_t() throw() { if(_p) (*pcre_free)(_p); }
51
52 pcre_t& operator=(pcre *p) { if(_p) (*pcre_free)(_p); _p=p; return *this; }
53
54 operator const pcre*(void) const { return _p; }
55 operator pcre*(void) { return _p; }
56
57 int exec(const string& s,pcre_matches_t& m) {
58 if(!_p)
59 throw internal_error(OPKELE_CP_ "Trying to execute absent regexp");
60 return pcre_exec(_p,NULL,s.c_str(),s.length(),0,0,m._ov,m._s);
61 }
62 };
63
26 class curl_t { 64 class curl_t {
27 public: 65 public:
28 CURL *_c; 66 CURL *_c;
29 67
30 curl_t() : _c(0) { } 68 curl_t() : _c(0) { }
31 curl_t(CURL *c) : _c(c) { } 69 curl_t(CURL *c) : _c(c) { }
32 ~curl_t() throw() { if(_c) curl_easy_cleanup(_c); } 70 ~curl_t() throw() { if(_c) curl_easy_cleanup(_c); }
33 71
34 curl_t& operator=(CURL *c) { if(_c) curl_easy_cleanup(_c); _c=c; return *this; } 72 curl_t& operator=(CURL *c) { if(_c) curl_easy_cleanup(_c); _c=c; return *this; }
35 73
36 operator const CURL*(void) const { return _c; } 74 operator const CURL*(void) const { return _c; }
37 operator CURL*(void) { return _c; } 75 operator CURL*(void) { return _c; }
38 }; 76 };
39 77
40 static CURLcode curl_misc_sets(CURL* c) { 78 static CURLcode curl_misc_sets(CURL* c) {
41 CURLcode r; 79 CURLcode r;
42 (r=curl_easy_setopt(c,CURLOPT_FOLLOWLOCATION,1)) 80 (r=curl_easy_setopt(c,CURLOPT_FOLLOWLOCATION,1))
43 || (r=curl_easy_setopt(c,CURLOPT_MAXREDIRS,5)) 81 || (r=curl_easy_setopt(c,CURLOPT_MAXREDIRS,5))
44 || (r=curl_easy_setopt(c,CURLOPT_DNS_CACHE_TIMEOUT,120)) 82 || (r=curl_easy_setopt(c,CURLOPT_DNS_CACHE_TIMEOUT,120))
45 || (r=curl_easy_setopt(c,CURLOPT_DNS_USE_GLOBAL_CACHE,1)) 83 || (r=curl_easy_setopt(c,CURLOPT_DNS_USE_GLOBAL_CACHE,1))
46 || (r=curl_easy_setopt(c,CURLOPT_USERAGENT,PACKAGE_NAME"/"PACKAGE_VERSION)) 84 || (r=curl_easy_setopt(c,CURLOPT_USERAGENT,PACKAGE_NAME"/"PACKAGE_VERSION))
47 || (r=curl_easy_setopt(c,CURLOPT_TIMEOUT,20)) 85 || (r=curl_easy_setopt(c,CURLOPT_TIMEOUT,20))
48 #ifdefDISABLE_CURL_SSL_VERIFYHOST 86 #ifdefDISABLE_CURL_SSL_VERIFYHOST
49 || (r=curl_easy_setopt(c,CURLOPT_SSL_VERIFYHOST,0)) 87 || (r=curl_easy_setopt(c,CURLOPT_SSL_VERIFYHOST,0))
50#endif 88#endif
51 #ifdefDISABLE_CURL_SSL_VERIFYPEER 89 #ifdefDISABLE_CURL_SSL_VERIFYPEER
52 || (r=curl_easy_setopt(c,CURLOPT_SSL_VERIFYPEER,0)) 90 || (r=curl_easy_setopt(c,CURLOPT_SSL_VERIFYPEER,0))
53#endif 91#endif
54 ; 92 ;
55 return r; 93 return r;
56 } 94 }
57 95
58 static size_t _curl_tostring(void *ptr,size_t size,size_t nmemb,void *stream) { 96 static size_t _curl_tostring(void *ptr,size_t size,size_t nmemb,void *stream) {
59 string *str = (string*)stream; 97 string *str = (string*)stream;
60 size_t bytes = size*nmemb; 98 size_t bytes = size*nmemb;
61 size_t get = min(16384-str->length(),bytes); 99 size_t get = min(16384-str->length(),bytes);
62 str->append((const char*)ptr,get); 100 str->append((const char*)ptr,get);
63 return get; 101 return get;
64 } 102 }
65 103
66 assoc_t consumer_t::associate(const string& server) { 104 assoc_t consumer_t::associate(const string& server) {
67 util::dh_t dh = DH_new(); 105 util::dh_t dh = DH_new();
68 if(!dh) 106 if(!dh)
69 throw exception_openssl(OPKELE_CP_ "failed to DH_new()"); 107 throw exception_openssl(OPKELE_CP_ "failed to DH_new()");
70 dh->p = util::dec_to_bignum(data::_default_p); 108 dh->p = util::dec_to_bignum(data::_default_p);
71 dh->g = util::dec_to_bignum(data::_default_g); 109 dh->g = util::dec_to_bignum(data::_default_g);
72 if(!DH_generate_key(dh)) 110 if(!DH_generate_key(dh))
73 throw exception_openssl(OPKELE_CP_ "failed to DH_generate_key()"); 111 throw exception_openssl(OPKELE_CP_ "failed to DH_generate_key()");
74 string request = 112 string request =
75 "openid.mode=associate" 113 "openid.mode=associate"
76 "&openid.assoc_type=HMAC-SHA1" 114 "&openid.assoc_type=HMAC-SHA1"
77 "&openid.session_type=DH-SHA1" 115 "&openid.session_type=DH-SHA1"
78 "&openid.dh_consumer_public="; 116 "&openid.dh_consumer_public=";
79 request += util::url_encode(util::bignum_to_base64(dh->pub_key)); 117 request += util::url_encode(util::bignum_to_base64(dh->pub_key));
80 curl_t curl = curl_easy_init(); 118 curl_t curl = curl_easy_init();
81 if(!curl) 119 if(!curl)
82 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()"); 120 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()");
83 string response; 121 string response;
84 CURLcode r; 122 CURLcode r;
85 (r=curl_misc_sets(curl)) 123 (r=curl_misc_sets(curl))
86 || (r=curl_easy_setopt(curl,CURLOPT_URL,server.c_str())) 124 || (r=curl_easy_setopt(curl,CURLOPT_URL,server.c_str()))
87 || (r=curl_easy_setopt(curl,CURLOPT_POST,1)) 125 || (r=curl_easy_setopt(curl,CURLOPT_POST,1))
88 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDS,request.data())) 126 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDS,request.data()))
89 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDSIZE,request.length())) 127 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDSIZE,request.length()))
90 || (r=curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,_curl_tostring)) 128 || (r=curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,_curl_tostring))
91 || (r=curl_easy_setopt(curl,CURLOPT_WRITEDATA,&response)) 129 || (r=curl_easy_setopt(curl,CURLOPT_WRITEDATA,&response))
92 ; 130 ;
93 if(r) 131 if(r)
94 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r); 132 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r);
95 if(r=curl_easy_perform(curl)) 133 if(r=curl_easy_perform(curl))
96 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r); 134 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r);
97 params_t p; p.parse_keyvalues(response); 135 params_t p; p.parse_keyvalues(response);
98 if(p.has_param("assoc_type") && p.get_param("assoc_type")!="HMAC-SHA1") 136 if(p.has_param("assoc_type") && p.get_param("assoc_type")!="HMAC-SHA1")
99 throw bad_input(OPKELE_CP_ "unsupported assoc_type"); 137 throw bad_input(OPKELE_CP_ "unsupported assoc_type");
100 string st; 138 string st;
101 if(p.has_param("session_type")) st = p.get_param("session_type"); 139 if(p.has_param("session_type")) st = p.get_param("session_type");
102 if((!st.empty()) && st!="DH-SHA1") 140 if((!st.empty()) && st!="DH-SHA1")
103 throw bad_input(OPKELE_CP_ "unsupported session_type"); 141 throw bad_input(OPKELE_CP_ "unsupported session_type");
104 secret_t secret; 142 secret_t secret;
105 if(st.empty()) { 143 if(st.empty()) {
106 secret.from_base64(p.get_param("mac_key")); 144 secret.from_base64(p.get_param("mac_key"));
107 }else{ 145 }else{
108 util::bignum_t s_pub = util::base64_to_bignum(p.get_param("dh_server_public")); 146 util::bignum_t s_pub = util::base64_to_bignum(p.get_param("dh_server_public"));
109 vector<unsigned char> ck(DH_size(dh)+1); 147 vector<unsigned char> ck(DH_size(dh)+1);
110 unsigned char *ckptr = &(ck.front())+1; 148 unsigned char *ckptr = &(ck.front())+1;
111 int cklen = DH_compute_key(ckptr,s_pub,dh); 149 int cklen = DH_compute_key(ckptr,s_pub,dh);
112 if(cklen<0) 150 if(cklen<0)
113 throw exception_openssl(OPKELE_CP_ "failed to DH_compute_key()"); 151 throw exception_openssl(OPKELE_CP_ "failed to DH_compute_key()");
114 if(cklen && (*ckptr)&0x80) { 152 if(cklen && (*ckptr)&0x80) {
115 (*(--ckptr)) = 0; ++cklen; 153 (*(--ckptr)) = 0; ++cklen;
116 } 154 }
117 unsigned char key_sha1[SHA_DIGEST_LENGTH]; 155 unsigned char key_sha1[SHA_DIGEST_LENGTH];
118 SHA1(ckptr,cklen,key_sha1); 156 SHA1(ckptr,cklen,key_sha1);
119 secret.enxor_from_base64(key_sha1,p.get_param("enc_mac_key")); 157 secret.enxor_from_base64(key_sha1,p.get_param("enc_mac_key"));
120 } 158 }
121 int expires_in = 0; 159 int expires_in = 0;
122 if(p.has_param("expires_in")) { 160 if(p.has_param("expires_in")) {
123 expires_in = util::string_to_long(p.get_param("expires_in")); 161 expires_in = util::string_to_long(p.get_param("expires_in"));
124 }else if(p.has_param("issued") && p.has_param("expiry")) { 162 }else if(p.has_param("issued") && p.has_param("expiry")) {
125 expires_in = util::w3c_to_time(p.get_param("expiry"))-util::w3c_to_time(p.get_param("issued")); 163 expires_in = util::w3c_to_time(p.get_param("expiry"))-util::w3c_to_time(p.get_param("issued"));
126 }else 164 }else
127 throw bad_input(OPKELE_CP_ "no expiration information"); 165 throw bad_input(OPKELE_CP_ "no expiration information");
128 return store_assoc(server,p.get_param("assoc_handle"),secret,expires_in); 166 return store_assoc(server,p.get_param("assoc_handle"),secret,expires_in);
129 } 167 }
130 168
131 string consumer_t::checkid_immediate(const string& identity,const string& return_to,const string& trust_root,extension_t *ext) { 169 string consumer_t::checkid_immediate(const string& identity,const string& return_to,const string& trust_root,extension_t *ext) {
132 return checkid_(mode_checkid_immediate,identity,return_to,trust_root,ext); 170 return checkid_(mode_checkid_immediate,identity,return_to,trust_root,ext);
133 } 171 }
134 string consumer_t::checkid_setup(const string& identity,const string& return_to,const string& trust_root,extension_t *ext) { 172 string consumer_t::checkid_setup(const string& identity,const string& return_to,const string& trust_root,extension_t *ext) {
135 return checkid_(mode_checkid_setup,identity,return_to,trust_root,ext); 173 return checkid_(mode_checkid_setup,identity,return_to,trust_root,ext);
136 } 174 }
137 string consumer_t::checkid_(mode_t mode,const string& identity,const string& return_to,const string& trust_root,extension_t *ext) { 175 string consumer_t::checkid_(mode_t mode,const string& identity,const string& return_to,const string& trust_root,extension_t *ext) {
138 params_t p; 176 params_t p;
139 if(mode==mode_checkid_immediate) 177 if(mode==mode_checkid_immediate)
140 p["mode"]="checkid_immediate"; 178 p["mode"]="checkid_immediate";
141 else if(mode==mode_checkid_setup) 179 else if(mode==mode_checkid_setup)
142 p["mode"]="checkid_setup"; 180 p["mode"]="checkid_setup";
143 else 181 else
144 throw bad_input(OPKELE_CP_ "unknown checkid_* mode"); 182 throw bad_input(OPKELE_CP_ "unknown checkid_* mode");
145 string iurl = canonicalize(identity); 183 string iurl = canonicalize(identity);
146 string server, delegate; 184 string server, delegate;
147 retrieve_links(iurl,server,delegate); 185 retrieve_links(iurl,server,delegate);
148 p["identity"] = delegate.empty()?iurl:delegate; 186 p["identity"] = delegate.empty()?iurl:delegate;
149 if(!trust_root.empty()) 187 if(!trust_root.empty())
150 p["trust_root"] = trust_root; 188 p["trust_root"] = trust_root;
151 p["return_to"] = return_to; 189 p["return_to"] = return_to;
152 try { 190 try {
153 string ah = find_assoc(server)->handle(); 191 string ah = find_assoc(server)->handle();
154 p["assoc_handle"] = ah; 192 p["assoc_handle"] = ah;
155 }catch(failed_lookup& fl) { 193 }catch(failed_lookup& fl) {
156 string ah = associate(server)->handle(); 194 string ah = associate(server)->handle();
157 p["assoc_handle"] = ah; 195 p["assoc_handle"] = ah;
158 } 196 }
159 if(ext) ext->checkid_hook(p,identity); 197 if(ext) ext->checkid_hook(p,identity);
160 return p.append_query(server); 198 return p.append_query(server);
161 } 199 }
162 200
163 void consumer_t::id_res(const params_t& pin,const string& identity,extension_t *ext) { 201 void consumer_t::id_res(const params_t& pin,const string& identity,extension_t *ext) {
164 if(pin.has_param("openid.user_setup_url")) 202 if(pin.has_param("openid.user_setup_url"))
165 throw id_res_setup(OPKELE_CP_ "assertion failed, setup url provided",pin.get_param("openid.user_setup_url")); 203 throw id_res_setup(OPKELE_CP_ "assertion failed, setup url provided",pin.get_param("openid.user_setup_url"));
166 string server,delegate; 204 string server,delegate;
167 retrieve_links(identity.empty()?pin.get_param("openid.identity"):canonicalize(identity),server,delegate); 205 retrieve_links(identity.empty()?pin.get_param("openid.identity"):canonicalize(identity),server,delegate);
168 params_t ps; 206 params_t ps;
169 try { 207 try {
170 assoc_t assoc = retrieve_assoc(server,pin.get_param("openid.assoc_handle")); 208 assoc_t assoc = retrieve_assoc(server,pin.get_param("openid.assoc_handle"));
171 if(assoc->is_expired()) /* TODO: or should I throw some other exception to force programmer fix his implementation? */ 209 if(assoc->is_expired()) /* TODO: or should I throw some other exception to force programmer fix his implementation? */
172 throw failed_lookup(OPKELE_CP_ "retrieve_assoc() has returned expired handle"); 210 throw failed_lookup(OPKELE_CP_ "retrieve_assoc() has returned expired handle");
173 const string& sigenc = pin.get_param("openid.sig"); 211 const string& sigenc = pin.get_param("openid.sig");
174 vector<unsigned char> sig; 212 vector<unsigned char> sig;
175 util::decode_base64(sigenc,sig); 213 util::decode_base64(sigenc,sig);
176 const string& slist = pin.get_param("openid.signed"); 214 const string& slist = pin.get_param("openid.signed");
177 string kv; 215 string kv;
178 string::size_type p = 0; 216 string::size_type p = 0;
179 while(true) { 217 while(true) {
180 string::size_type co = slist.find(',',p); 218 string::size_type co = slist.find(',',p);
181 string f = (co==string::npos)?slist.substr(p):slist.substr(p,co-p); 219 string f = (co==string::npos)?slist.substr(p):slist.substr(p,co-p);
182 kv += f; 220 kv += f;
183 kv += ':'; 221 kv += ':';
184 f.insert(0,"openid."); 222 f.insert(0,"openid.");
185 kv += pin.get_param(f); 223 kv += pin.get_param(f);
186 kv += '\n'; 224 kv += '\n';
187 if(ext) ps[f.substr(sizeof("openid.")-1)] = pin.get_param(f); 225 if(ext) ps[f.substr(sizeof("openid.")-1)] = pin.get_param(f);
188 if(co==string::npos) 226 if(co==string::npos)
189 break; 227 break;
190 p = co+1; 228 p = co+1;
191 } 229 }
192 secret_t secret = assoc->secret(); 230 secret_t secret = assoc->secret();
193 unsigned int md_len = 0; 231 unsigned int md_len = 0;
194 unsigned char *md = HMAC( 232 unsigned char *md = HMAC(
195 EVP_sha1(), 233 EVP_sha1(),
196 &(secret.front()),secret.size(), 234 &(secret.front()),secret.size(),
197 (const unsigned char *)kv.data(),kv.length(), 235 (const unsigned char *)kv.data(),kv.length(),
198 0,&md_len); 236 0,&md_len);
199 if(sig.size()!=md_len || memcmp(&(sig.front()),md,md_len)) 237 if(sig.size()!=md_len || memcmp(&(sig.front()),md,md_len))
200 throw id_res_mismatch(OPKELE_CP_ "signature mismatch"); 238 throw id_res_mismatch(OPKELE_CP_ "signature mismatch");
201 }catch(failed_lookup& e) { /* XXX: more specific? */ 239 }catch(failed_lookup& e) { /* XXX: more specific? */
202 const string& slist = pin.get_param("openid.signed"); 240 const string& slist = pin.get_param("openid.signed");
203 string::size_type pp = 0; 241 string::size_type pp = 0;
204 params_t p; 242 params_t p;
205 while(true) { 243 while(true) {
206 string::size_type co = slist.find(',',pp); 244 string::size_type co = slist.find(',',pp);
207 string f = "openid."; 245 string f = "openid.";
208 f += (co==string::npos)?slist.substr(pp):slist.substr(pp,co-pp); 246 f += (co==string::npos)?slist.substr(pp):slist.substr(pp,co-pp);
209 p[f] = pin.get_param(f); 247 p[f] = pin.get_param(f);
210 if(co==string::npos) 248 if(co==string::npos)
211 break; 249 break;
212 pp = co+1; 250 pp = co+1;
213 } 251 }
214 p["openid.assoc_handle"] = pin.get_param("openid.assoc_handle"); 252 p["openid.assoc_handle"] = pin.get_param("openid.assoc_handle");
215 p["openid.sig"] = pin.get_param("openid.sig"); 253 p["openid.sig"] = pin.get_param("openid.sig");
216 p["openid.signed"] = pin.get_param("openid.signed"); 254 p["openid.signed"] = pin.get_param("openid.signed");
217 try { 255 try {
218 string ih = pin.get_param("openid.invalidate_handle"); 256 string ih = pin.get_param("openid.invalidate_handle");
219 p["openid.invalidate_handle"] = ih; 257 p["openid.invalidate_handle"] = ih;
220 }catch(failed_lookup& fl) { } 258 }catch(failed_lookup& fl) { }
221 try { 259 try {
222 check_authentication(server,p); 260 check_authentication(server,p);
223 }catch(failed_check_authentication& fca) { 261 }catch(failed_check_authentication& fca) {
224 throw id_res_failed(OPKELE_CP_ "failed to check_authentication()"); 262 throw id_res_failed(OPKELE_CP_ "failed to check_authentication()");
225 } 263 }
226 } 264 }
227 if(ext) ext->id_res_hook(pin,ps,identity); 265 if(ext) ext->id_res_hook(pin,ps,identity);
228 } 266 }
229 267
230 void consumer_t::check_authentication(const string& server,const params_t& p) { 268 void consumer_t::check_authentication(const string& server,const params_t& p) {
231 string request = "openid.mode=check_authentication"; 269 string request = "openid.mode=check_authentication";
232 for(params_t::const_iterator i=p.begin();i!=p.end();++i) { 270 for(params_t::const_iterator i=p.begin();i!=p.end();++i) {
233 if(i->first!="openid.mode") { 271 if(i->first!="openid.mode") {
234 request += '&'; 272 request += '&';
235 request += i->first; 273 request += i->first;
236 request += '='; 274 request += '=';
237 request += util::url_encode(i->second); 275 request += util::url_encode(i->second);
238 } 276 }
239 } 277 }
240 curl_t curl = curl_easy_init(); 278 curl_t curl = curl_easy_init();
241 if(!curl) 279 if(!curl)
242 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()"); 280 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()");
243 string response; 281 string response;
244 CURLcode r; 282 CURLcode r;
245 (r=curl_misc_sets(curl)) 283 (r=curl_misc_sets(curl))
246 || (r=curl_easy_setopt(curl,CURLOPT_URL,server.c_str())) 284 || (r=curl_easy_setopt(curl,CURLOPT_URL,server.c_str()))
247 || (r=curl_easy_setopt(curl,CURLOPT_POST,1)) 285 || (r=curl_easy_setopt(curl,CURLOPT_POST,1))
248 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDS,request.data())) 286 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDS,request.data()))
249 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDSIZE,request.length())) 287 || (r=curl_easy_setopt(curl,CURLOPT_POSTFIELDSIZE,request.length()))
250 || (r=curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,_curl_tostring)) 288 || (r=curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,_curl_tostring))
251 || (r=curl_easy_setopt(curl,CURLOPT_WRITEDATA,&response)) 289 || (r=curl_easy_setopt(curl,CURLOPT_WRITEDATA,&response))
252 ; 290 ;
253 if(r) 291 if(r)
254 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r); 292 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r);
255 if(r=curl_easy_perform(curl)) 293 if(r=curl_easy_perform(curl))
256 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r); 294 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r);
257 params_t pp; pp.parse_keyvalues(response); 295 params_t pp; pp.parse_keyvalues(response);
258 if(pp.has_param("invalidate_handle")) 296 if(pp.has_param("invalidate_handle"))
259 invalidate_assoc(server,pp.get_param("invalidate_handle")); 297 invalidate_assoc(server,pp.get_param("invalidate_handle"));
260 if(pp.has_param("is_valid")) { 298 if(pp.has_param("is_valid")) {
261 if(pp.get_param("is_valid")=="true") 299 if(pp.get_param("is_valid")=="true")
262 return; 300 return;
263 }else if(pp.has_param("lifetime")) { 301 }else if(pp.has_param("lifetime")) {
264 if(util::string_to_long(pp.get_param("lifetime"))) 302 if(util::string_to_long(pp.get_param("lifetime")))
265 return; 303 return;
266 } 304 }
267 throw failed_check_authentication(OPKELE_CP_ "failed to verify response"); 305 throw failed_check_authentication(OPKELE_CP_ "failed to verify response");
268 } 306 }
269 307
270 void consumer_t::retrieve_links(const string& url,string& server,string& delegate) { 308 void consumer_t::retrieve_links(const string& url,string& server,string& delegate) {
271#if defined(USE_LIBPCRECPP) || defined(USE_PCREPP)
272 server.erase(); 309 server.erase();
273 delegate.erase(); 310 delegate.erase();
274 curl_t curl = curl_easy_init(); 311 curl_t curl = curl_easy_init();
275 if(!curl) 312 if(!curl)
276 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()"); 313 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()");
277 string html; 314 string html;
278 CURLcode r; 315 CURLcode r;
279 (r=curl_misc_sets(curl)) 316 (r=curl_misc_sets(curl))
280 || (r=curl_easy_setopt(curl,CURLOPT_URL,url.c_str())) 317 || (r=curl_easy_setopt(curl,CURLOPT_URL,url.c_str()))
281 || (r=curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,_curl_tostring)) 318 || (r=curl_easy_setopt(curl,CURLOPT_WRITEFUNCTION,_curl_tostring))
282 || (r=curl_easy_setopt(curl,CURLOPT_WRITEDATA,&html)) 319 || (r=curl_easy_setopt(curl,CURLOPT_WRITEDATA,&html))
283 ; 320 ;
284 if(r) 321 if(r)
285 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r); 322 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r);
286 r = curl_easy_perform(curl); 323 r = curl_easy_perform(curl);
287 if(r && r!=CURLE_WRITE_ERROR) 324 if(r && r!=CURLE_WRITE_ERROR)
288 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r); 325 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r);
289 // strip out everything past body 326 static const char *re_bre = "<\\s*body\\b", *re_hdre = "<\\s*head[^>]*>",
290 static const char *re_hdre = "<\\s*head[^>]*>",
291 *re_lre = "<\\s*link\\b([^>]+)>", 327 *re_lre = "<\\s*link\\b([^>]+)>",
292 *re_rre = "\\brel\\s*=\\s*['\"]\\s*([^'\"\\s]+)\\s*['\"]", 328 *re_rre = "\\brel\\s*=\\s*['\"]\\s*([^'\"\\s]+)\\s*['\"]",
293 *re_hre = "\\bhref\\s*=\\s*['\"]\\s*([^'\"\\s]+)\\s*['\"]"; 329 *re_hre = "\\bhref\\s*=\\s*['\"]\\s*([^'\"\\s]+)\\s*['\"]";
294#if defined(USE_LIBPCRECPP) 330 pcre_matches_t m1(3), m2(3);
295 static pcrecpp::RE_Options ro(PCRE_CASELESS|PCRE_DOTALL); 331 pcre_t bre(re_bre,PCRE_CASELESS);
296 static pcrecpp::RE 332 if(bre.exec(html,m1)>0)
297 bre("<body\\b.*",ro), hdre(re_hdre,ro), 333 html.erase(m1.begin(0));
298 lre(re_lre,ro), rre(re_rre), hre(re_hre,ro); 334 pcre_t hdre(re_hdre,PCRE_CASELESS);
299 bre.Replace("",&html); 335 if(hdre.exec(html,m1)<=0)
300 pcrecpp::StringPiece hpiece(html); 336 throw bad_input(OPKELE_CP_ "failed to find <head>");
301 if(!hdre.FindAndConsume(&hpiece)) 337 html.erase(0,m1.end(0)+1);
302 throw bad_input(OPKELE_CP_ "failed to find head"); 338 pcre_t lre(re_lre,PCRE_CASELESS), rre(re_rre,PCRE_CASELESS), hre(re_hre,PCRE_CASELESS);
303 string attrs; 339 while(lre.exec(html,m1)>=2) {
304 while(lre.FindAndConsume(&hpiece,&attrs)) { 340 string attrs(html,m1.begin(1),m1.length(1));
305 pcrecpp::StringPiece rel, href; 341 html.erase(0,m1.end(0)+1);
306 if(!(rre.PartialMatch(attrs,&rel) && hre.PartialMatch(attrs,&href))) 342 if(!( rre.exec(attrs,m1)>=2 && hre.exec(attrs,m2)>=2 ))
307 continue; 343 continue;
344 string rel(attrs,m1.begin(1),m1.length(1));
345 string href(attrs,m2.begin(1),m2.length(1));
308 if(rel=="openid.server") { 346 if(rel=="openid.server") {
309 href.CopyToString(&server); 347 server = href;
310 if(!delegate.empty()) 348 if(!delegate.empty()) break;
311 break;
312 }else if(rel=="openid.delegate") { 349 }else if(rel=="openid.delegate") {
313 href.CopyToString(&delegate); 350 delegate = href;
314 if(!server.empty()) 351 if(!server.empty()) break;
315 break;
316 }
317 }
318#elif defined(USE_PCREPP)
319 pcrepp::Pcre bre("<body\\b",PCRE_CASELESS);
320 if(bre.search(html))
321 html.erase(bre.get_match_start());
322 pcrepp::Pcre hdre(re_hdre,PCRE_CASELESS);
323 if(!hdre.search(html))
324 throw bad_input(OPKELE_CP_ "failed to find head");
325 html.erase(0,hdre.get_match_end()+1);
326 pcrepp::Pcre lre(re_lre,PCRE_CASELESS), rre(re_rre,PCRE_CASELESS), hre(re_hre,PCRE_CASELESS);
327 while(lre.search(html)) {
328 string attrs = lre[0];
329 html.erase(0,lre.get_match_end()+1);
330 if(!(rre.search(attrs)&&hre.search(attrs)))
331 continue;
332 if(rre[0]=="openid.server") {
333 server = hre[0];
334 if(!delegate.empty())
335 break;
336 }else if(rre[0]=="openid.delegate") {
337 delegate = hre[0];
338 if(!server.empty())
339 break;
340 } 352 }
341 } 353 }
342#else
343 #error "I must have gone crazy"
344#endif
345 if(server.empty()) 354 if(server.empty())
346 throw failed_assertion(OPKELE_CP_ "The location has no openid.server declaration"); 355 throw failed_assertion(OPKELE_CP_ "The location has no openid.server declaration");
347#else /* none of the RE bindings enabled */
348 throw not_implemented(OPKELE_CP_ "No internal implementation of retrieve_links were provided at compile-time");
349#endif
350 } 356 }
351 357
352 assoc_t consumer_t::find_assoc(const string& server) { 358 assoc_t consumer_t::find_assoc(const string& server) {
353 throw failed_lookup(OPKELE_CP_ "no find_assoc() provided"); 359 throw failed_lookup(OPKELE_CP_ "no find_assoc() provided");
354 } 360 }
355 361
356 string consumer_t::normalize(const string& url) { 362 string consumer_t::normalize(const string& url) {
357 string rv = url; 363 string rv = url;
358 // strip leading and trailing spaces 364 // strip leading and trailing spaces
359 string::size_type i = rv.find_first_not_of(" \t\r\n"); 365 string::size_type i = rv.find_first_not_of(" \t\r\n");
360 if(i==string::npos) 366 if(i==string::npos)
361 throw bad_input(OPKELE_CP_ "empty URL"); 367 throw bad_input(OPKELE_CP_ "empty URL");
362 if(i) 368 if(i)
363 rv.erase(0,i); 369 rv.erase(0,i);
364 i = rv.find_last_not_of(" \t\r\n"); 370 i = rv.find_last_not_of(" \t\r\n");
365 assert(i!=string::npos); 371 assert(i!=string::npos);
366 if(i<(rv.length()-1)) 372 if(i<(rv.length()-1))
367 rv.erase(i+1); 373 rv.erase(i+1);
368 // add missing http:// 374 // add missing http://
369 i = rv.find("://"); 375 i = rv.find("://");
370 if(i==string::npos) { // primitive. but do we need more? 376 if(i==string::npos) { // primitive. but do we need more?
371 rv.insert(0,"http://"); 377 rv.insert(0,"http://");
372 i = sizeof("http://")-1; 378 i = sizeof("http://")-1;
373 }else{ 379 }else{
374 i += sizeof("://")-1; 380 i += sizeof("://")-1;
375 } 381 }
376 string::size_type qm = rv.find('?',i); 382 string::size_type qm = rv.find('?',i);
377 string::size_type sl = rv.find('/',i); 383 string::size_type sl = rv.find('/',i);
378 if(qm!=string::npos) { 384 if(qm!=string::npos) {
379 if(sl==string::npos || sl>qm) 385 if(sl==string::npos || sl>qm)
380 rv.insert(qm,1,'/'); 386 rv.insert(qm,1,'/');
381 }else{ 387 }else{
382 if(sl==string::npos) 388 if(sl==string::npos)
383 rv += '/'; 389 rv += '/';
384 } 390 }
385 return rv; 391 return rv;
386 } 392 }
387 393
388 string consumer_t::canonicalize(const string& url) { 394 string consumer_t::canonicalize(const string& url) {
389 string rv = normalize(url); 395 string rv = normalize(url);
390 curl_t curl = curl_easy_init(); 396 curl_t curl = curl_easy_init();
391 if(!curl) 397 if(!curl)
392 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()"); 398 throw exception_curl(OPKELE_CP_ "failed to curl_easy_init()");
393 string html; 399 string html;
394 CURLcode r; 400 CURLcode r;
395 (r=curl_misc_sets(curl)) 401 (r=curl_misc_sets(curl))
396 || (r=curl_easy_setopt(curl,CURLOPT_URL,rv.c_str())) 402 || (r=curl_easy_setopt(curl,CURLOPT_URL,rv.c_str()))
397 || (r=curl_easy_setopt(curl,CURLOPT_NOBODY,1)) 403 || (r=curl_easy_setopt(curl,CURLOPT_NOBODY,1))
398 ; 404 ;
399 if(r) 405 if(r)
400 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r); 406 throw exception_curl(OPKELE_CP_ "failed to curl_easy_setopt()",r);
401 r = curl_easy_perform(curl); 407 r = curl_easy_perform(curl);
402 if(r) 408 if(r)
403 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r); 409 throw exception_curl(OPKELE_CP_ "failed to curl_easy_perform()",r);
404 const char *eu = 0; 410 const char *eu = 0;
405 r = curl_easy_getinfo(curl,CURLINFO_EFFECTIVE_URL,&eu); 411 r = curl_easy_getinfo(curl,CURLINFO_EFFECTIVE_URL,&eu);
406 if(r) 412 if(r)
407 throw exception_curl(OPKELE_CP_ "failed to curl_easy_getinfo(..CURLINFO_EFFECTIVE_URL..)",r); 413 throw exception_curl(OPKELE_CP_ "failed to curl_easy_getinfo(..CURLINFO_EFFECTIVE_URL..)",r);
408 rv = eu; 414 rv = eu;
409 return normalize(rv); 415 return normalize(rv);
410 } 416 }
411 417
412} 418}
diff --git a/libopkele.pc.in b/libopkele.pc.in
index 286720e..5f8684b 100644
--- a/libopkele.pc.in
+++ b/libopkele.pc.in
@@ -1,11 +1,11 @@
1prefix=@prefix@ 1prefix=@prefix@
2exec_prefix=@exec_prefix@ 2exec_prefix=@exec_prefix@
3libdir=@libdir@ 3libdir=@libdir@
4includedir=@includedir@ 4includedir=@includedir@
5 5
6Name: libopkele 6Name: libopkele
7Description: C++ implementation of OpenID protocol 7Description: C++ implementation of OpenID protocol
8Version: @VERSION@ 8Version: @VERSION@
9Requires: openssl @KONFORKA_KONFORKA@ 9Requires: openssl libpcre @KONFORKA_KONFORKA@
10Cflags: -I${includedir} @LIBCURL_CPPFLAGS@ @PCREPP_CFLAGS@ 10Cflags: -I${includedir} @LIBCURL_CPPFLAGS@ @PCREPP_CFLAGS@
11Libs: -L${libdir} -lopkele @LIBCURL@ @PCREPP_LIBS@ 11Libs: -L${libdir} -lopkele @LIBCURL@ @PCREPP_LIBS@