| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-09-15 15:44:26 +00:00
										 |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2006-02-12 04:28:58 +00:00
										 |  |  |  * Copyright (C)  2004 - 2006, Tilghman Lesher | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  |  * Tilghman Lesher <curl-20050919@the-tilghman.com> | 
					
						
							| 
									
										
										
										
											2005-01-05 06:38:39 +00:00
										 |  |  |  * and Brian Wilkins <bwilkins@cfl.rr.com> (Added POST option) | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * app_curl.c is distributed with no restrictions on usage or | 
					
						
							|  |  |  |  * redistribution. | 
					
						
							| 
									
										
										
										
											2005-09-15 15:44:26 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * See http://www.asterisk.org for more information about
 | 
					
						
							|  |  |  |  * the Asterisk project. Please do not directly contact | 
					
						
							|  |  |  |  * any of the maintainers of this project for assistance; | 
					
						
							|  |  |  |  * the project provides a web site, mailing lists and IRC | 
					
						
							|  |  |  |  * channels for your use. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  | /*! \file
 | 
					
						
							| 
									
										
										
										
											2005-12-30 21:18:06 +00:00
										 |  |  |  *  | 
					
						
							| 
									
										
										
										
											2006-02-23 22:27:49 +00:00
										 |  |  |  * \brief Curl - Load a URL | 
					
						
							| 
									
										
										
										
											2005-12-30 21:18:06 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \author Tilghman Lesher <curl-20050919@the-tilghman.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note Brian Wilkins <bwilkins@cfl.rr.com> (Added POST option)  | 
					
						
							| 
									
										
										
										
											2007-02-24 19:27:50 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \extref Depends on the CURL library  - http://curl.haxx.se/
 | 
					
						
							| 
									
										
										
										
											2005-09-15 15:44:26 +00:00
										 |  |  |  *  | 
					
						
							| 
									
										
										
										
											2006-02-23 22:27:49 +00:00
										 |  |  |  * \ingroup functions | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  |   | 
					
						
							| 
									
										
										
										
											2006-04-24 17:11:45 +00:00
										 |  |  | /*** MODULEINFO
 | 
					
						
							|  |  |  | 	<depend>curl</depend> | 
					
						
							| 
									
										
										
										
											2011-07-14 20:28:54 +00:00
										 |  |  | 	<support_level>core</support_level> | 
					
						
							| 
									
										
										
										
											2006-04-24 17:11:45 +00:00
										 |  |  |  ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-07 18:54:56 +00:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-06-06 22:39:32 +00:00
										 |  |  | #include <curl/curl.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-21 06:02:45 +00:00
										 |  |  | #include "asterisk/lock.h"
 | 
					
						
							|  |  |  | #include "asterisk/file.h"
 | 
					
						
							|  |  |  | #include "asterisk/channel.h"
 | 
					
						
							|  |  |  | #include "asterisk/pbx.h"
 | 
					
						
							|  |  |  | #include "asterisk/cli.h"
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							| 
									
										
										
										
											2005-11-30 19:38:54 +00:00
										 |  |  | #include "asterisk/app.h"
 | 
					
						
							| 
									
										
										
										
											2006-01-11 22:41:34 +00:00
										 |  |  | #include "asterisk/utils.h"
 | 
					
						
							| 
									
										
										
										
											2006-12-16 04:25:46 +00:00
										 |  |  | #include "asterisk/threadstorage.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-28 17:34:57 +00:00
										 |  |  | #include "asterisk/test.h"
 | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | /*** DOCUMENTATION
 | 
					
						
							|  |  |  | 	<function name="CURL" language="en_US"> | 
					
						
							|  |  |  | 		<synopsis> | 
					
						
							|  |  |  | 			Retrieve content from a remote web or ftp server | 
					
						
							|  |  |  | 		</synopsis> | 
					
						
							|  |  |  | 		<syntax> | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 			<parameter name="url" required="true"> | 
					
						
							|  |  |  | 				<para>The full URL for the resource to retrieve.</para> | 
					
						
							|  |  |  | 			</parameter> | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 			<parameter name="post-data"> | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 				<para><emphasis>Read Only</emphasis></para> | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 				<para>If specified, an <literal>HTTP POST</literal> will be | 
					
						
							|  |  |  | 				performed with the content of | 
					
						
							|  |  |  | 				<replaceable>post-data</replaceable>, instead of an | 
					
						
							|  |  |  | 				<literal>HTTP GET</literal> (default).</para> | 
					
						
							|  |  |  | 			</parameter> | 
					
						
							|  |  |  | 		</syntax> | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 		<description> | 
					
						
							|  |  |  | 			<para>When this function is read, a <literal>HTTP GET</literal> | 
					
						
							|  |  |  | 			(by default) will be used to retrieve the contents of the provided | 
					
						
							|  |  |  | 			<replaceable>url</replaceable>. The contents are returned as the | 
					
						
							|  |  |  | 			result of the function.</para> | 
					
						
							|  |  |  | 			<example title="Displaying contents of a page" language="text"> | 
					
						
							|  |  |  | 			exten => s,1,Verbose(0, ${CURL(http://localhost:8088/static/astman.css)})
 | 
					
						
							|  |  |  | 			</example> | 
					
						
							|  |  |  | 			<para>When this function is written to, a <literal>HTTP GET</literal> | 
					
						
							|  |  |  | 			will be used to retrieve the contents of the provided | 
					
						
							|  |  |  | 			<replaceable>url</replaceable>. The value written to the function | 
					
						
							|  |  |  | 			specifies the destination file of the cURL'd resource.</para> | 
					
						
							|  |  |  | 			<example title="Retrieving a file" language="text"> | 
					
						
							|  |  |  | 			exten => s,1,Set(CURL(http://localhost:8088/static/astman.css)=/var/spool/asterisk/tmp/astman.css))
 | 
					
						
							|  |  |  | 			</example> | 
					
						
							|  |  |  | 			<note> | 
					
						
							|  |  |  | 				<para>If <literal>live_dangerously</literal> in <literal>asterisk.conf</literal> | 
					
						
							|  |  |  | 				is set to <literal>no</literal>, this function can only be written to from the | 
					
						
							|  |  |  | 				dialplan, and not directly from external protocols. Read operations are | 
					
						
							|  |  |  | 				unaffected.</para> | 
					
						
							|  |  |  | 			</note> | 
					
						
							|  |  |  | 		</description> | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 		<see-also> | 
					
						
							|  |  |  | 			<ref type="function">CURLOPT</ref> | 
					
						
							|  |  |  | 		</see-also> | 
					
						
							|  |  |  | 	</function> | 
					
						
							|  |  |  | 	<function name="CURLOPT" language="en_US"> | 
					
						
							|  |  |  | 		<synopsis> | 
					
						
							| 
									
										
										
										
											2011-01-06 17:50:57 +00:00
										 |  |  | 			Sets various options for future invocations of CURL. | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 		</synopsis> | 
					
						
							|  |  |  | 		<syntax> | 
					
						
							|  |  |  | 			<parameter name="key" required="yes"> | 
					
						
							|  |  |  | 				<enumlist> | 
					
						
							|  |  |  | 					<enum name="cookie"> | 
					
						
							|  |  |  | 						<para>A cookie to send with the request.  Multiple | 
					
						
							|  |  |  | 						cookies are supported.</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="conntimeout"> | 
					
						
							|  |  |  | 						<para>Number of seconds to wait for a connection to succeed</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="dnstimeout"> | 
					
						
							|  |  |  | 						<para>Number of seconds to wait for DNS to be resolved</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="ftptext"> | 
					
						
							|  |  |  | 						<para>For FTP URIs, force a text transfer (boolean)</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="ftptimeout"> | 
					
						
							|  |  |  | 						<para>For FTP URIs, number of seconds to wait for a | 
					
						
							|  |  |  | 						server response</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="header"> | 
					
						
							|  |  |  | 						<para>Include header information in the result | 
					
						
							|  |  |  | 						(boolean)</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="httptimeout"> | 
					
						
							|  |  |  | 						<para>For HTTP(S) URIs, number of seconds to wait for a | 
					
						
							|  |  |  | 						server response</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="maxredirs"> | 
					
						
							|  |  |  | 						<para>Maximum number of redirects to follow</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="proxy"> | 
					
						
							|  |  |  | 						<para>Hostname or IP address to use as a proxy server</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="proxytype"> | 
					
						
							|  |  |  | 						<para>Type of <literal>proxy</literal></para> | 
					
						
							|  |  |  | 						<enumlist> | 
					
						
							|  |  |  | 							<enum name="http" /> | 
					
						
							|  |  |  | 							<enum name="socks4" /> | 
					
						
							|  |  |  | 							<enum name="socks5" /> | 
					
						
							|  |  |  | 						</enumlist> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="proxyport"> | 
					
						
							|  |  |  | 						<para>Port number of the <literal>proxy</literal></para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="proxyuserpwd"> | 
					
						
							|  |  |  | 						<para>A <replaceable>username</replaceable><literal>:</literal><replaceable>password</replaceable> | 
					
						
							|  |  |  | 						combination to use for authenticating requests through a | 
					
						
							|  |  |  | 						<literal>proxy</literal></para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="referer"> | 
					
						
							|  |  |  | 						<para>Referer URL to use for the request</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="useragent"> | 
					
						
							|  |  |  | 						<para>UserAgent string to use for the request</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="userpwd"> | 
					
						
							|  |  |  | 						<para>A <replaceable>username</replaceable><literal>:</literal><replaceable>password</replaceable> | 
					
						
							|  |  |  | 						to use for authentication when the server response to | 
					
						
							|  |  |  | 						an initial request indicates a 401 status code.</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="ssl_verifypeer"> | 
					
						
							|  |  |  | 						<para>Whether to verify the server certificate against | 
					
						
							|  |  |  | 						a list of known root certificate authorities (boolean).</para> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 					<enum name="hashcompat"> | 
					
						
							|  |  |  | 						<para>Assuming the responses will be in <literal>key1=value1&key2=value2</literal> | 
					
						
							|  |  |  | 						format, reformat the response such that it can be used | 
					
						
							|  |  |  | 						by the <literal>HASH</literal> function.</para> | 
					
						
							|  |  |  | 						<enumlist> | 
					
						
							|  |  |  | 							<enum name="yes" /> | 
					
						
							|  |  |  | 							<enum name="no" /> | 
					
						
							|  |  |  | 							<enum name="legacy"> | 
					
						
							| 
									
										
										
										
											2011-01-06 17:50:57 +00:00
										 |  |  | 								<para>Also translate <literal>+</literal> to the | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 								space character, in violation of current RFC | 
					
						
							|  |  |  | 								standards.</para> | 
					
						
							|  |  |  | 							</enum> | 
					
						
							|  |  |  | 						</enumlist> | 
					
						
							|  |  |  | 					</enum> | 
					
						
							|  |  |  | 				</enumlist> | 
					
						
							|  |  |  | 			</parameter> | 
					
						
							|  |  |  | 		</syntax> | 
					
						
							|  |  |  | 		<description> | 
					
						
							|  |  |  | 			<para>Options may be set globally or per channel.  Per-channel | 
					
						
							|  |  |  | 			settings will override global settings.</para> | 
					
						
							|  |  |  | 		</description> | 
					
						
							|  |  |  | 		<see-also> | 
					
						
							|  |  |  | 			<ref type="function">CURL</ref> | 
					
						
							|  |  |  | 			<ref type="function">HASH</ref> | 
					
						
							|  |  |  | 		</see-also> | 
					
						
							|  |  |  | 	</function> | 
					
						
							|  |  |  |  ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | #define CURLVERSION_ATLEAST(a,b,c) \
 | 
					
						
							|  |  |  | 	((LIBCURL_VERSION_MAJOR > (a)) || ((LIBCURL_VERSION_MAJOR == (a)) && (LIBCURL_VERSION_MINOR > (b))) || ((LIBCURL_VERSION_MAJOR == (a)) && (LIBCURL_VERSION_MINOR == (b)) && (LIBCURL_VERSION_PATCH >= (c)))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-09 12:57:21 +00:00
										 |  |  | #define CURLOPT_SPECIAL_HASHCOMPAT ((CURLoption) -500)
 | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | static void curlds_free(void *data); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-18 17:18:20 +00:00
										 |  |  | static const struct ast_datastore_info curl_info = { | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	.type = "CURL", | 
					
						
							|  |  |  | 	.destroy = curlds_free, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct curl_settings { | 
					
						
							|  |  |  | 	AST_LIST_ENTRY(curl_settings) list; | 
					
						
							|  |  |  | 	CURLoption key; | 
					
						
							|  |  |  | 	void *value; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AST_LIST_HEAD_STATIC(global_curl_info, curl_settings); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void curlds_free(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	AST_LIST_HEAD(global_curl_info, curl_settings) *list = data; | 
					
						
							| 
									
										
										
										
											2008-10-08 12:15:06 +00:00
										 |  |  | 	struct curl_settings *setting; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	if (!list) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-10-08 12:15:06 +00:00
										 |  |  | 	while ((setting = AST_LIST_REMOVE_HEAD(list, list))) { | 
					
						
							| 
									
										
										
										
											2014-09-26 14:41:38 +00:00
										 |  |  | 		ast_free(setting); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	AST_LIST_HEAD_DESTROY(list); | 
					
						
							| 
									
										
										
										
											2015-01-12 15:18:24 +00:00
										 |  |  | 	ast_free(list); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum optiontype { | 
					
						
							|  |  |  | 	OT_BOOLEAN, | 
					
						
							|  |  |  | 	OT_INTEGER, | 
					
						
							|  |  |  | 	OT_INTEGER_MS, | 
					
						
							|  |  |  | 	OT_STRING, | 
					
						
							|  |  |  | 	OT_ENUM, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | enum hashcompat { | 
					
						
							|  |  |  | 	HASHCOMPAT_NO = 0, | 
					
						
							|  |  |  | 	HASHCOMPAT_YES, | 
					
						
							|  |  |  | 	HASHCOMPAT_LEGACY, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | static int parse_curlopt_key(const char *name, CURLoption *key, enum optiontype *ot) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!strcasecmp(name, "header")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_HEADER; | 
					
						
							|  |  |  | 		*ot = OT_BOOLEAN; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "proxy")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_PROXY; | 
					
						
							|  |  |  | 		*ot = OT_STRING; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "proxyport")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_PROXYPORT; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "proxytype")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_PROXYTYPE; | 
					
						
							|  |  |  | 		*ot = OT_ENUM; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "dnstimeout")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_DNS_CACHE_TIMEOUT; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "userpwd")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_USERPWD; | 
					
						
							|  |  |  | 		*ot = OT_STRING; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "proxyuserpwd")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_PROXYUSERPWD; | 
					
						
							|  |  |  | 		*ot = OT_STRING; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "maxredirs")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_MAXREDIRS; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "referer")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_REFERER; | 
					
						
							|  |  |  | 		*ot = OT_STRING; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "useragent")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_USERAGENT; | 
					
						
							|  |  |  | 		*ot = OT_STRING; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "cookie")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_COOKIE; | 
					
						
							|  |  |  | 		*ot = OT_STRING; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "ftptimeout")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_FTP_RESPONSE_TIMEOUT; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER; | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "httptimeout")) { | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,16,2)
 | 
					
						
							|  |  |  | 		*key = CURLOPT_TIMEOUT_MS; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER_MS; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		*key = CURLOPT_TIMEOUT; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "conntimeout")) { | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,16,2)
 | 
					
						
							|  |  |  | 		*key = CURLOPT_CONNECTTIMEOUT_MS; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER_MS; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		*key = CURLOPT_CONNECTTIMEOUT; | 
					
						
							|  |  |  | 		*ot = OT_INTEGER; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} else if (!strcasecmp(name, "ftptext")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_TRANSFERTEXT; | 
					
						
							|  |  |  | 		*ot = OT_BOOLEAN; | 
					
						
							| 
									
										
										
										
											2009-08-14 17:36:40 +00:00
										 |  |  | 	} else if (!strcasecmp(name, "ssl_verifypeer")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_SSL_VERIFYPEER; | 
					
						
							|  |  |  | 		*ot = OT_BOOLEAN; | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 	} else if (!strcasecmp(name, "hashcompat")) { | 
					
						
							|  |  |  | 		*key = CURLOPT_SPECIAL_HASHCOMPAT; | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 		*ot = OT_ENUM; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int acf_curlopt_write(struct ast_channel *chan, const char *cmd, char *name, const char *value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_datastore *store; | 
					
						
							| 
									
										
										
										
											2008-09-06 15:40:15 +00:00
										 |  |  | 	struct global_curl_info *list; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	struct curl_settings *cur, *new = NULL; | 
					
						
							|  |  |  | 	CURLoption key; | 
					
						
							|  |  |  | 	enum optiontype ot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (chan) { | 
					
						
							|  |  |  | 		if (!(store = ast_channel_datastore_find(chan, &curl_info, NULL))) { | 
					
						
							|  |  |  | 			/* Create a new datastore */ | 
					
						
							|  |  |  | 			if (!(store = ast_datastore_alloc(&curl_info, NULL))) { | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR, "Unable to allocate new datastore.  Cannot set any CURL options\n"); | 
					
						
							|  |  |  | 				return -1; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (!(list = ast_calloc(1, sizeof(*list)))) { | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR, "Unable to allocate list head.  Cannot set any CURL options\n"); | 
					
						
							|  |  |  | 				ast_datastore_free(store); | 
					
						
							| 
									
										
										
										
											2012-04-09 21:47:54 +00:00
										 |  |  | 				return -1; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			store->data = list; | 
					
						
							|  |  |  | 			AST_LIST_HEAD_INIT(list); | 
					
						
							|  |  |  | 			ast_channel_datastore_add(chan, store); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			list = store->data; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* Populate the global structure */ | 
					
						
							| 
									
										
										
										
											2008-09-06 15:40:15 +00:00
										 |  |  | 		list = &global_curl_info; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!parse_curlopt_key(name, &key, &ot)) { | 
					
						
							|  |  |  | 		if (ot == OT_BOOLEAN) { | 
					
						
							|  |  |  | 			if ((new = ast_calloc(1, sizeof(*new)))) { | 
					
						
							| 
									
										
										
										
											2008-09-05 22:03:26 +00:00
										 |  |  | 				new->value = (void *)((long) ast_true(value)); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} else if (ot == OT_INTEGER) { | 
					
						
							|  |  |  | 			long tmp = atol(value); | 
					
						
							|  |  |  | 			if ((new = ast_calloc(1, sizeof(*new)))) { | 
					
						
							|  |  |  | 				new->value = (void *)tmp; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else if (ot == OT_INTEGER_MS) { | 
					
						
							|  |  |  | 			long tmp = atof(value) * 1000.0; | 
					
						
							|  |  |  | 			if ((new = ast_calloc(1, sizeof(*new)))) { | 
					
						
							|  |  |  | 				new->value = (void *)tmp; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else if (ot == OT_STRING) { | 
					
						
							|  |  |  | 			if ((new = ast_calloc(1, sizeof(*new) + strlen(value) + 1))) { | 
					
						
							|  |  |  | 				new->value = (char *)new + sizeof(*new); | 
					
						
							|  |  |  | 				strcpy(new->value, value); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else if (ot == OT_ENUM) { | 
					
						
							|  |  |  | 			if (key == CURLOPT_PROXYTYPE) { | 
					
						
							|  |  |  | 				long ptype = | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,10,0)
 | 
					
						
							|  |  |  | 					CURLPROXY_HTTP; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 					CURLPROXY_SOCKS5; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 				if (0) { | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,15,2)
 | 
					
						
							|  |  |  | 				} else if (!strcasecmp(value, "socks4")) { | 
					
						
							|  |  |  | 					ptype = CURLPROXY_SOCKS4; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,18,0)
 | 
					
						
							|  |  |  | 				} else if (!strcasecmp(value, "socks4a")) { | 
					
						
							|  |  |  | 					ptype = CURLPROXY_SOCKS4A; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,18,0)
 | 
					
						
							|  |  |  | 				} else if (!strcasecmp(value, "socks5")) { | 
					
						
							|  |  |  | 					ptype = CURLPROXY_SOCKS5; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,18,0)
 | 
					
						
							|  |  |  | 				} else if (!strncasecmp(value, "socks5", 6)) { | 
					
						
							|  |  |  | 					ptype = CURLPROXY_SOCKS5_HOSTNAME; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if ((new = ast_calloc(1, sizeof(*new)))) { | 
					
						
							|  |  |  | 					new->value = (void *)ptype; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 			} else if (key == CURLOPT_SPECIAL_HASHCOMPAT) { | 
					
						
							|  |  |  | 				if ((new = ast_calloc(1, sizeof(*new)))) { | 
					
						
							|  |  |  | 					new->value = (void *) (long) (!strcasecmp(value, "legacy") ? HASHCOMPAT_LEGACY : ast_true(value) ? HASHCOMPAT_YES : HASHCOMPAT_NO); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				/* Highly unlikely */ | 
					
						
							|  |  |  | 				goto yuck; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Memory allocation error */ | 
					
						
							|  |  |  | 		if (!new) { | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		new->key = key; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | yuck: | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Unrecognized option: %s\n", name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Remove any existing entry */ | 
					
						
							|  |  |  | 	AST_LIST_LOCK(list); | 
					
						
							|  |  |  | 	AST_LIST_TRAVERSE_SAFE_BEGIN(list, cur, list) { | 
					
						
							|  |  |  | 		if (cur->key == new->key) { | 
					
						
							|  |  |  | 			AST_LIST_REMOVE_CURRENT(list); | 
					
						
							| 
									
										
										
										
											2014-09-26 14:41:38 +00:00
										 |  |  | 			ast_free(cur); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	AST_LIST_TRAVERSE_SAFE_END | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Insert new entry */ | 
					
						
							|  |  |  | 	ast_debug(1, "Inserting entry %p with key %d and value %p\n", new, new->key, new->value); | 
					
						
							|  |  |  | 	AST_LIST_INSERT_TAIL(list, new, list); | 
					
						
							|  |  |  | 	AST_LIST_UNLOCK(list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | static int acf_curlopt_helper(struct ast_channel *chan, const char *cmd, char *data, char *buf, struct ast_str **bufstr, ssize_t len) | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_datastore *store; | 
					
						
							| 
									
										
										
										
											2008-09-06 15:40:15 +00:00
										 |  |  | 	struct global_curl_info *list[2] = { &global_curl_info, NULL }; | 
					
						
							| 
									
										
										
										
											2008-09-07 00:04:05 +00:00
										 |  |  | 	struct curl_settings *cur = NULL; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	CURLoption key; | 
					
						
							|  |  |  | 	enum optiontype ot; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (parse_curlopt_key(data, &key, &ot)) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Unrecognized option: '%s'\n", data); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (chan && (store = ast_channel_datastore_find(chan, &curl_info, NULL))) { | 
					
						
							|  |  |  | 		list[0] = store->data; | 
					
						
							| 
									
										
										
										
											2008-09-06 15:40:15 +00:00
										 |  |  | 		list[1] = &global_curl_info; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < 2; i++) { | 
					
						
							|  |  |  | 		if (!list[i]) { | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		AST_LIST_LOCK(list[i]); | 
					
						
							|  |  |  | 		AST_LIST_TRAVERSE(list[i], cur, list) { | 
					
						
							|  |  |  | 			if (cur->key == key) { | 
					
						
							|  |  |  | 				if (ot == OT_BOOLEAN || ot == OT_INTEGER) { | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | 					if (buf) { | 
					
						
							|  |  |  | 						snprintf(buf, len, "%ld", (long) cur->value); | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						ast_str_set(bufstr, len, "%ld", (long) cur->value); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 				} else if (ot == OT_INTEGER_MS) { | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | 					if ((long) cur->value % 1000 == 0) { | 
					
						
							|  |  |  | 						if (buf) { | 
					
						
							|  |  |  | 							snprintf(buf, len, "%ld", (long)cur->value / 1000); | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							ast_str_set(bufstr, len, "%ld", (long) cur->value / 1000); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 					} else { | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | 						if (buf) { | 
					
						
							|  |  |  | 							snprintf(buf, len, "%.3f", (double) ((long) cur->value) / 1000.0); | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							ast_str_set(bufstr, len, "%.3f", (double) ((long) cur->value) / 1000.0); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} else if (ot == OT_STRING) { | 
					
						
							|  |  |  | 					ast_debug(1, "Found entry %p, with key %d and value %p\n", cur, cur->key, cur->value); | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | 					if (buf) { | 
					
						
							|  |  |  | 						ast_copy_string(buf, cur->value, len); | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						ast_str_set(bufstr, 0, "%s", (char *) cur->value); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 				} else if (key == CURLOPT_PROXYTYPE) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 					const char *strval = "unknown"; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 					if (0) { | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,15,2)
 | 
					
						
							|  |  |  | 					} else if ((long)cur->value == CURLPROXY_SOCKS4) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 						strval = "socks4"; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,18,0)
 | 
					
						
							|  |  |  | 					} else if ((long)cur->value == CURLPROXY_SOCKS4A) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 						strval = "socks4a"; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 					} else if ((long)cur->value == CURLPROXY_SOCKS5) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 						strval = "socks5"; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | #if CURLVERSION_ATLEAST(7,18,0)
 | 
					
						
							|  |  |  | 					} else if ((long)cur->value == CURLPROXY_SOCKS5_HOSTNAME) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 						strval = "socks5hostname"; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #if CURLVERSION_ATLEAST(7,10,0)
 | 
					
						
							|  |  |  | 					} else if ((long)cur->value == CURLPROXY_HTTP) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 						strval = "http"; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 					if (buf) { | 
					
						
							|  |  |  | 						ast_copy_string(buf, strval, len); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 					} else { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 						ast_str_set(bufstr, 0, "%s", strval); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} else if (key == CURLOPT_SPECIAL_HASHCOMPAT) { | 
					
						
							|  |  |  | 					const char *strval = "unknown"; | 
					
						
							|  |  |  | 					if ((long) cur->value == HASHCOMPAT_LEGACY) { | 
					
						
							|  |  |  | 						strval = "legacy"; | 
					
						
							|  |  |  | 					} else if ((long) cur->value == HASHCOMPAT_YES) { | 
					
						
							|  |  |  | 						strval = "yes"; | 
					
						
							|  |  |  | 					} else if ((long) cur->value == HASHCOMPAT_NO) { | 
					
						
							|  |  |  | 						strval = "no"; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					if (buf) { | 
					
						
							|  |  |  | 						ast_copy_string(buf, strval, len); | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						ast_str_set(bufstr, 0, "%s", strval); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		AST_LIST_UNLOCK(list[i]); | 
					
						
							|  |  |  | 		if (cur) { | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return cur ? 0 : -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | static int acf_curlopt_read(struct ast_channel *chan, const char *cmd, char *data, char *buf, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return acf_curlopt_helper(chan, cmd, data, buf, NULL, len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int acf_curlopt_read2(struct ast_channel *chan, const char *cmd, char *data, struct ast_str **buf, ssize_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return acf_curlopt_helper(chan, cmd, data, NULL, buf, len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | /*! \brief Callback data passed to \ref WriteMemoryCallback */ | 
					
						
							|  |  |  | struct curl_write_callback_data { | 
					
						
							|  |  |  | 	/*! \brief If a string is being built, the string buffer */ | 
					
						
							|  |  |  | 	struct ast_str *str; | 
					
						
							|  |  |  | 	/*! \brief The max size of \ref str */ | 
					
						
							|  |  |  | 	ssize_t len; | 
					
						
							|  |  |  | 	/*! \brief If a file is being retrieved, the file to write to */ | 
					
						
							|  |  |  | 	FILE *out_file; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | static size_t WriteMemoryCallback(void *ptr, size_t size, size_t nmemb, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	register int realsize = 0; | 
					
						
							|  |  |  | 	struct curl_write_callback_data *cb_data = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (cb_data->str) { | 
					
						
							|  |  |  | 		realsize = size * nmemb; | 
					
						
							|  |  |  | 		ast_str_append_substr(&cb_data->str, 0, ptr, realsize); | 
					
						
							|  |  |  | 	} else if (cb_data->out_file) { | 
					
						
							|  |  |  | 		realsize = fwrite(ptr, size, nmemb, cb_data->out_file); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | 	return realsize; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-12 13:59:35 +00:00
										 |  |  | static const char * const global_useragent = "asterisk-libcurl-agent/1.0"; | 
					
						
							| 
									
										
										
										
											2006-12-16 04:25:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-12-16 21:24:08 +00:00
										 |  |  | static int curl_instance_init(void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CURL **curl = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(*curl = curl_easy_init())) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	curl_easy_setopt(*curl, CURLOPT_NOSIGNAL, 1); | 
					
						
							|  |  |  | 	curl_easy_setopt(*curl, CURLOPT_TIMEOUT, 180); | 
					
						
							|  |  |  | 	curl_easy_setopt(*curl, CURLOPT_WRITEFUNCTION, WriteMemoryCallback); | 
					
						
							|  |  |  | 	curl_easy_setopt(*curl, CURLOPT_USERAGENT, global_useragent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-12-16 04:25:46 +00:00
										 |  |  | static void curl_instance_cleanup(void *data) | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-12-16 04:25:46 +00:00
										 |  |  | 	CURL **curl = data; | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-12-16 04:25:46 +00:00
										 |  |  | 	curl_easy_cleanup(*curl); | 
					
						
							| 
									
										
										
										
											2008-07-30 19:55:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ast_free(data); | 
					
						
							| 
									
										
										
										
											2006-12-16 04:25:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AST_THREADSTORAGE_CUSTOM(curl_instance, curl_instance_init, curl_instance_cleanup); | 
					
						
							| 
									
										
										
										
											2011-09-21 20:53:13 +00:00
										 |  |  | AST_THREADSTORAGE(thread_escapebuf); | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-28 17:34:57 +00:00
										 |  |  | /*!
 | 
					
						
							|  |  |  |  * \brief Check for potential HTTP injection risk. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * CVE-2014-8150 brought up the fact that HTTP proxies are subject to injection | 
					
						
							|  |  |  |  * attacks. An HTTP URL sent to a proxy contains a carriage-return linefeed combination, | 
					
						
							|  |  |  |  * followed by a complete HTTP request. Proxies will handle this as two separate HTTP | 
					
						
							|  |  |  |  * requests rather than as a malformed URL. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * libcURL patched this vulnerability in version 7.40.0, but we have no guarantee that | 
					
						
							|  |  |  |  * Asterisk systems will be using an up-to-date cURL library. Therefore, we implement | 
					
						
							|  |  |  |  * the same fix as libcURL for determining if a URL is vulnerable to an injection attack. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param url The URL to check for vulnerability | 
					
						
							|  |  |  |  * \retval 0 The URL is not vulnerable | 
					
						
							|  |  |  |  * \retval 1 The URL is vulnerable. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int url_is_vulnerable(const char *url) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (strpbrk(url, "\r\n")) { | 
					
						
							|  |  |  | 		return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | struct curl_args { | 
					
						
							|  |  |  | 	const char *url; | 
					
						
							|  |  |  | 	const char *postdata; | 
					
						
							|  |  |  | 	struct curl_write_callback_data cb_data; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int acf_curl_helper(struct ast_channel *chan, struct curl_args *args) | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-09-21 20:53:13 +00:00
										 |  |  | 	struct ast_str *escapebuf = ast_str_thread_get(&thread_escapebuf, 16); | 
					
						
							| 
									
										
										
										
											2008-10-06 21:09:05 +00:00
										 |  |  | 	int ret = -1; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	CURL **curl; | 
					
						
							|  |  |  | 	struct curl_settings *cur; | 
					
						
							|  |  |  | 	struct ast_datastore *store = NULL; | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 	int hashcompat = 0; | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	AST_LIST_HEAD(global_curl_info, curl_settings) *list = NULL; | 
					
						
							| 
									
										
										
										
											2013-03-20 20:27:37 +00:00
										 |  |  | 	char curl_errbuf[CURL_ERROR_SIZE + 1]; /* add one to be safe */ | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-21 20:53:13 +00:00
										 |  |  | 	if (!escapebuf) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	if (!(curl = ast_threadstorage_get(&curl_instance, sizeof(*curl)))) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Cannot allocate curl structure\n"); | 
					
						
							| 
									
										
										
										
											2006-02-12 04:28:58 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-01-05 06:38:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	if (url_is_vulnerable(args->url)) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "URL '%s' is vulnerable to HTTP injection attacks. Aborting CURL() call.\n", args->url); | 
					
						
							| 
									
										
										
										
											2015-01-28 17:34:57 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	if (chan) { | 
					
						
							| 
									
										
										
										
											2007-10-26 14:00:48 +00:00
										 |  |  | 		ast_autoservice_start(chan); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2007-09-19 19:36:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	AST_LIST_LOCK(&global_curl_info); | 
					
						
							|  |  |  | 	AST_LIST_TRAVERSE(&global_curl_info, cur, list) { | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 		if (cur->key == CURLOPT_SPECIAL_HASHCOMPAT) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 			hashcompat = (long) cur->value; | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			curl_easy_setopt(*curl, cur->key, cur->value); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-14 02:01:12 +00:00
										 |  |  | 	AST_LIST_UNLOCK(&global_curl_info); | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	if (chan && (store = ast_channel_datastore_find(chan, &curl_info, NULL))) { | 
					
						
							|  |  |  | 		list = store->data; | 
					
						
							|  |  |  | 		AST_LIST_LOCK(list); | 
					
						
							|  |  |  | 		AST_LIST_TRAVERSE(list, cur, list) { | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 			if (cur->key == CURLOPT_SPECIAL_HASHCOMPAT) { | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | 				hashcompat = (long) cur->value; | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				curl_easy_setopt(*curl, cur->key, cur->value); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2005-05-26 16:15:58 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	curl_easy_setopt(*curl, CURLOPT_URL, args->url); | 
					
						
							|  |  |  | 	curl_easy_setopt(*curl, CURLOPT_FILE, (void *) &args->cb_data); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	if (args->postdata) { | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 		curl_easy_setopt(*curl, CURLOPT_POST, 1); | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 		curl_easy_setopt(*curl, CURLOPT_POSTFIELDS, args->postdata); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-20 20:27:37 +00:00
										 |  |  | 	/* Temporarily assign a buffer for curl to write errors to. */ | 
					
						
							|  |  |  | 	curl_errbuf[0] = curl_errbuf[CURL_ERROR_SIZE] = '\0'; | 
					
						
							|  |  |  | 	curl_easy_setopt(*curl, CURLOPT_ERRORBUFFER, curl_errbuf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (curl_easy_perform(*curl) != 0) { | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "%s ('%s')\n", curl_errbuf, args->url); | 
					
						
							| 
									
										
										
										
											2013-03-20 20:27:37 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Reset buffer to NULL so curl doesn't try to write to it when the
 | 
					
						
							|  |  |  | 	 * buffer is deallocated. Documentation is vague about allowing NULL | 
					
						
							|  |  |  | 	 * here, but the source allows it. See: "typecheck: allow NULL to unset | 
					
						
							|  |  |  | 	 * CURLOPT_ERRORBUFFER" (62bcf005f4678a93158358265ba905bace33b834). */ | 
					
						
							|  |  |  | 	curl_easy_setopt(*curl, CURLOPT_ERRORBUFFER, (char*)NULL); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (store) { | 
					
						
							|  |  |  | 		AST_LIST_UNLOCK(list); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	if (args->postdata) { | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 		curl_easy_setopt(*curl, CURLOPT_POST, 0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	if (args->cb_data.str && ast_str_strlen(args->cb_data.str)) { | 
					
						
							|  |  |  | 		ast_str_trim_blanks(args->cb_data.str); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 		ast_debug(3, "CURL returned str='%s'\n", ast_str_buffer(args->cb_data.str)); | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 		if (hashcompat) { | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 			char *remainder = ast_str_buffer(args->cb_data.str); | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 			char *piece; | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 			struct ast_str *fields = ast_str_create(ast_str_strlen(args->cb_data.str) / 2); | 
					
						
							|  |  |  | 			struct ast_str *values = ast_str_create(ast_str_strlen(args->cb_data.str) / 2); | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 			int rowcount = 0; | 
					
						
							| 
									
										
										
										
											2010-11-15 07:45:42 +00:00
										 |  |  | 			while (fields && values && (piece = strsep(&remainder, "&"))) { | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 				char *name = strsep(&piece, "="); | 
					
						
							| 
									
										
										
										
											2013-03-20 20:18:40 +00:00
										 |  |  | 				struct ast_flags mode = (hashcompat == HASHCOMPAT_LEGACY ? ast_uri_http_legacy : ast_uri_http); | 
					
						
							|  |  |  | 				if (piece) { | 
					
						
							|  |  |  | 					ast_uri_decode(piece, mode); | 
					
						
							| 
									
										
										
										
											2011-01-07 18:23:52 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2013-03-20 20:18:40 +00:00
										 |  |  | 				ast_uri_decode(name, mode); | 
					
						
							| 
									
										
										
										
											2011-09-21 20:53:13 +00:00
										 |  |  | 				ast_str_append(&fields, 0, "%s%s", rowcount ? "," : "", ast_str_set_escapecommas(&escapebuf, 0, name, INT_MAX)); | 
					
						
							|  |  |  | 				ast_str_append(&values, 0, "%s%s", rowcount ? "," : "", ast_str_set_escapecommas(&escapebuf, 0, S_OR(piece, ""), INT_MAX)); | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 				rowcount++; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-12-13 08:36:35 +00:00
										 |  |  | 			pbx_builtin_setvar_helper(chan, "~ODBCFIELDS~", ast_str_buffer(fields)); | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 			ast_str_set(&args->cb_data.str, 0, "%s", ast_str_buffer(values)); | 
					
						
							| 
									
										
										
										
											2008-09-24 06:43:05 +00:00
										 |  |  | 			ast_free(fields); | 
					
						
							|  |  |  | 			ast_free(values); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-10-06 21:09:05 +00:00
										 |  |  | 		ret = 0; | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	if (chan) { | 
					
						
							| 
									
										
										
										
											2007-10-26 14:00:48 +00:00
										 |  |  | 		ast_autoservice_stop(chan); | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-06 21:09:05 +00:00
										 |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | static int acf_curl_exec(struct ast_channel *chan, const char *cmd, char *info, struct ast_str **buf, ssize_t len) | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	struct curl_args curl_params = { 0, }; | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_DECLARE_APP_ARGS(args, | 
					
						
							|  |  |  | 		AST_APP_ARG(url); | 
					
						
							|  |  |  | 		AST_APP_ARG(postdata); | 
					
						
							|  |  |  | 	); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_STANDARD_APP_ARGS(args, info); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_strlen_zero(info)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "CURL requires an argument (URL)\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	curl_params.url = args.url; | 
					
						
							|  |  |  | 	curl_params.postdata = args.postdata; | 
					
						
							|  |  |  | 	curl_params.cb_data.str = ast_str_create(16); | 
					
						
							|  |  |  | 	if (!curl_params.cb_data.str) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res = acf_curl_helper(chan, &curl_params); | 
					
						
							|  |  |  | 	ast_str_set(buf, len, "%s", ast_str_buffer(curl_params.cb_data.str)); | 
					
						
							|  |  |  | 	ast_free(curl_params.cb_data.str); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | static int acf_curl_write(struct ast_channel *chan, const char *cmd, char *name, const char *value) | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	struct curl_args curl_params = { 0, }; | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	char *args_value = ast_strdupa(value); | 
					
						
							|  |  |  | 	AST_DECLARE_APP_ARGS(args, | 
					
						
							|  |  |  | 		AST_APP_ARG(file_path); | 
					
						
							|  |  |  | 	); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_STANDARD_APP_ARGS(args, args_value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_strlen_zero(name)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "CURL requires an argument (URL)\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_strlen_zero(args.file_path)) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "CURL requires a file to write\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	curl_params.url = name; | 
					
						
							|  |  |  | 	curl_params.cb_data.out_file = fopen(args.file_path, "w"); | 
					
						
							|  |  |  | 	if (!curl_params.cb_data.out_file) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Failed to open file %s: %s (%d)\n", | 
					
						
							|  |  |  | 			args.file_path, | 
					
						
							|  |  |  | 			strerror(errno), | 
					
						
							|  |  |  | 			errno); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res = acf_curl_helper(chan, &curl_params); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fclose(curl_params.cb_data.out_file); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-15 17:34:30 +00:00
										 |  |  | static struct ast_custom_function acf_curl = { | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | 	.name = "CURL", | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	.read2 = acf_curl_exec, | 
					
						
							|  |  |  | 	.write = acf_curl_write, | 
					
						
							| 
									
										
										
										
											2005-09-25 19:36:48 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-15 17:34:30 +00:00
										 |  |  | static struct ast_custom_function acf_curlopt = { | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	.name = "CURLOPT", | 
					
						
							|  |  |  | 	.synopsis = "Set options for use with the CURL() function", | 
					
						
							|  |  |  | 	.syntax = "CURLOPT(<option>)", | 
					
						
							|  |  |  | 	.desc = | 
					
						
							| 
									
										
										
										
											2009-08-14 17:36:40 +00:00
										 |  |  | "  cookie         - Send cookie with request [none]\n" | 
					
						
							|  |  |  | "  conntimeout    - Number of seconds to wait for connection\n" | 
					
						
							|  |  |  | "  dnstimeout     - Number of seconds to wait for DNS response\n" | 
					
						
							|  |  |  | "  ftptext        - For FTP, force a text transfer (boolean)\n" | 
					
						
							|  |  |  | "  ftptimeout     - For FTP, the server response timeout\n" | 
					
						
							|  |  |  | "  header         - Retrieve header information (boolean)\n" | 
					
						
							|  |  |  | "  httptimeout    - Number of seconds to wait for HTTP response\n" | 
					
						
							|  |  |  | "  maxredirs      - Maximum number of redirects to follow\n" | 
					
						
							|  |  |  | "  proxy          - Hostname or IP to use as a proxy\n" | 
					
						
							|  |  |  | "  proxytype      - http, socks4, or socks5\n" | 
					
						
							|  |  |  | "  proxyport      - port number of the proxy\n" | 
					
						
							|  |  |  | "  proxyuserpwd   - A <user>:<pass> to use for authentication\n" | 
					
						
							|  |  |  | "  referer        - Referer URL to use for the request\n" | 
					
						
							|  |  |  | "  useragent      - UserAgent string to use\n" | 
					
						
							|  |  |  | "  userpwd        - A <user>:<pass> to use for authentication\n" | 
					
						
							|  |  |  | "  ssl_verifypeer - Whether to verify the peer certificate (boolean)\n" | 
					
						
							|  |  |  | "  hashcompat     - Result data will be compatible for use with HASH()\n" | 
					
						
							| 
									
										
										
										
											2011-01-06 17:28:32 +00:00
										 |  |  | "                 - if value is \"legacy\", will translate '+' to ' '\n" | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | "", | 
					
						
							|  |  |  | 	.read = acf_curlopt_read, | 
					
						
							| 
									
										
										
										
											2009-04-29 18:53:01 +00:00
										 |  |  | 	.read2 = acf_curlopt_read2, | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	.write = acf_curlopt_write, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-18 19:40:22 -04:00
										 |  |  | #ifdef TEST_FRAMEWORK
 | 
					
						
							| 
									
										
										
										
											2015-01-28 17:34:57 +00:00
										 |  |  | AST_TEST_DEFINE(vulnerable_url) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const char *bad_urls [] = { | 
					
						
							|  |  |  | 		"http://example.com\r\nDELETE http://example.com/everything", | 
					
						
							|  |  |  | 		"http://example.com\rDELETE http://example.com/everything", | 
					
						
							|  |  |  | 		"http://example.com\nDELETE http://example.com/everything", | 
					
						
							|  |  |  | 		"\r\nhttp://example.com", | 
					
						
							|  |  |  | 		"\rhttp://example.com", | 
					
						
							|  |  |  | 		"\nhttp://example.com", | 
					
						
							|  |  |  | 		"http://example.com\r\n", | 
					
						
							|  |  |  | 		"http://example.com\r", | 
					
						
							|  |  |  | 		"http://example.com\n", | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	const char *good_urls [] = { | 
					
						
							|  |  |  | 		"http://example.com", | 
					
						
							|  |  |  | 		"http://example.com/%5Cr%5Cn", | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	enum ast_test_result_state res = AST_TEST_PASS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (cmd) { | 
					
						
							|  |  |  | 	case TEST_INIT: | 
					
						
							|  |  |  | 		info->name = "vulnerable_url"; | 
					
						
							|  |  |  | 		info->category = "/funcs/func_curl/"; | 
					
						
							|  |  |  | 		info->summary = "cURL vulnerable URL test"; | 
					
						
							|  |  |  | 		info->description = | 
					
						
							|  |  |  | 			"Ensure that any combination of '\\r' or '\\n' in a URL invalidates the URL"; | 
					
						
							|  |  |  | 	case TEST_EXECUTE: | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < ARRAY_LEN(bad_urls); ++i) { | 
					
						
							|  |  |  | 		if (!url_is_vulnerable(bad_urls[i])) { | 
					
						
							|  |  |  | 			ast_test_status_update(test, "String '%s' detected as valid when it should be invalid\n", bad_urls[i]); | 
					
						
							|  |  |  | 			res = AST_TEST_FAIL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < ARRAY_LEN(good_urls); ++i) { | 
					
						
							|  |  |  | 		if (url_is_vulnerable(good_urls[i])) { | 
					
						
							|  |  |  | 			ast_test_status_update(test, "String '%s' detected as invalid when it should be valid\n", good_urls[i]); | 
					
						
							|  |  |  | 			res = AST_TEST_FAIL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-07-18 19:40:22 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-01-28 17:34:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int unload_module(void) | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res = ast_custom_function_unregister(&acf_curl); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	res |= ast_custom_function_unregister(&acf_curlopt); | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-28 17:34:57 +00:00
										 |  |  | 	AST_TEST_UNREGISTER(vulnerable_url); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int load_module(void) | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-25 16:00:54 +00:00
										 |  |  | 	if (!ast_module_check("res_curl.so")) { | 
					
						
							|  |  |  | 		if (ast_load_resource("res_curl.so") != AST_MODULE_LOAD_SUCCESS) { | 
					
						
							|  |  |  | 			ast_log(LOG_ERROR, "Cannot load res_curl, so func_curl cannot be loaded\n"); | 
					
						
							|  |  |  | 			return AST_MODULE_LOAD_DECLINE; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-09-03 19:07:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 01:21:18 +00:00
										 |  |  | 	res = ast_custom_function_register_escalating(&acf_curl, AST_CFE_WRITE); | 
					
						
							| 
									
										
										
										
											2008-09-05 19:12:03 +00:00
										 |  |  | 	res |= ast_custom_function_register(&acf_curlopt); | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-28 17:34:57 +00:00
										 |  |  | 	AST_TEST_REGISTER(vulnerable_url); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-18 22:52:21 +00:00
										 |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2004-12-23 02:55:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-20 19:35:02 +00:00
										 |  |  | AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "Load external URL", | 
					
						
							| 
									
										
										
										
											2015-05-05 20:49:04 -04:00
										 |  |  | 	.support_level = AST_MODULE_SUPPORT_CORE, | 
					
						
							|  |  |  | 	.load = load_module, | 
					
						
							|  |  |  | 	.unload = unload_module, | 
					
						
							|  |  |  | 	.load_pri = AST_MODPRI_REALTIME_DEPEND2, | 
					
						
							|  |  |  | ); | 
					
						
							| 
									
										
										
										
											2006-04-24 17:11:45 +00:00
										 |  |  | 
 |