package SMS::Send::UK::Kapow; use warnings; use strict; use Carp; =head1 NAME SMS::Send::UK::Kapow - SMS::Send driver for the Kapow.co.uk website =cut use SMS::Send::Driver (); use base 'SMS::Send::Driver'; use version; our $VERSION = qv('0.06'); use URI::Escape; =head1 SYNOPSIS use SMS::Send; my $sender = SMS::Send->new('UK::Kapow', _login => 'my-kapow-username', # normally required, see below (synonymous with _user) _password => 'my-kapow-password', # normally required, see below _send_via => 'http', # optional, can be http, https or email, default is http _http_method => 'get', # optional, the http method to use for http & https. # get or post, default post. _email_via => 'sendmail', # optional, for use when 'email' is used. can be # 'sendmail' or 'smtp' _url => 'http://foo.com/done' # optional url to call after sending, for http and https _from => 'me@mydomain.com', # optional, for use when 'email' is used in send_via _from_id => 'my-kapow-id', # optional message originator, if enabled for your account _route => '840101', # optional shortcode for premium sms reverse billing _wait => 0, # optional, supply a defined but false value to stop the # module from trying to confirm delivery. default true. ); my $sent = $sender->send_sms( to => '447712345678', # the recipient phone number text => "Hello, world!", # the text of the message to send _url => 'http://foo.com/done123', # optional url per message to call after sending # (for http and https) _from => 'me@mydomain.com', # optional from address per message (for email) _from_id => 'my-kapow-id', # optional message originator per message _route => '840101', # optional shortcode per message for reverse billing _wait => 0, # optional per message control of delivery checks. ); # Did it send to Kapow ok? if ( $sent ) { print "Sent test message\n"; } else { print "Test message failed\n"; } # What's the delivery status of the last message we sent? (available for http & https methods) my $status = $sender->delivery_status; # What's the delivery status for an arbitrary message we sent in the past? my $status = $sender->delivery_status($sent); =head1 DESCRIPTION L is a L driver that delivers messages via the L website. Messages for any country can be sent through this interface, although Kapow is generally aimed at users in the UK. This driver is based on the Kapow implementation document available at L =head2 Preparing to Use This Driver You need to create an account at http://www.kapow.co.uk to be able to use this driver. You will also need to purchase some SMS credits for your account. Other optional services such getting a custom 'from-id' or setting trusted sender email addresses can be configured in your Kapow account and used through this module. =head1 METHODS =head2 new # Create a new sender using this driver my $sender = SMS::Send->new('UK::Kapow', _login => 'username', # normally required but see below _password => 'password', # normally required but see below ); In most cases you should provide your Kapow username and password. If, however, you have set your Kapow account to allow any SMS requests received via email from a certain trusted email address then you do not need to set your username and password. In that case you may need to provide a '_from' parameter here or in the send_sms method unless your user account's email address on your system is the same as the trusted one registered with your Kapow account. =over =item _login The C<_login> param should be your kapow.co.uk username. The C<_user> parameter can be used instead. =item _password The C<_password> param should be your kapow.co.uk password. =item _send_via The C<_send_via> param controls how your SMS messages are sent to Kapow. The default method is http, which issues an http request from your server to Kapow's. Other options are 'https' and 'email'. The preferred methods are 'http' or 'https' because they allow the system to track the delivery statuses of individual messages. If your server cannot issue http/s requests then you can set this to 'email' and it will instead send an email message to Kapow to issue the SMS message. Note that using email means you lose the delivery tracking features of the http/s methods. =item _http_method The C<_http_method> param is relevant when using http or https to send your messages and specifies whether to use 'get' or 'post' methods when issuing the request to Kapow. This defaults to 'post' but can be changed to 'get' if desired. =item _email_via The C<_email_via> param is relevant when using email to send the messages. Valid values for this param are 'sendmail' and 'smtp'. The default is 'sendmail' except on Windows, where the default is 'smtp'. =item _url The C<_url> param is relevant when using http or https to send your messages. It provides an arbitrary callback URL which the Kapow server should call once the message has been successfully delivered. You can override this per-message in order to use unique URLs which relate to specific messages. =item _from The C<_from> param is relevant when using email to send your messages and it should contain the email address to us as the 'from' address on the emails sent to Kapow. The default (if using email) is to use your user account's default email address. =item _from_id The C<_from_id> param is relevant for users who use Kapow's 'from-id' feature to control the sender information that the recipient sees. If you have purchased this feature from Kapow for your account then you can control it using this parameter. =item _route The C<_route> param is for users who have purchased Kapow's Premium SMS service to reverse-bill recipients for sending an SMS message. If you have purchased this feature from Kapow for your account then you can control it using this parameter. =item _wait The C<_wait> param is relevant when using http or https to send your messages. The SMS::Send specification dictates that if a message can have its delivery checked & confirmed then the driver module should do that automatically at the time the message is sent. This means that if you use http/s to send your messages then the system will wait for up to 10 seconds trying to validate that the message has gone. If you want to override this behaviour then supply a defined-but-false value for this param, such as 0 or the empty string. Note that if you do suppress the automatic checking as described above then you can still check the state of delivery of each message using the delivery_status() method. =back Returns a new C object, or dies on error. =cut use LWP::UserAgent; use MIME::Lite; my $IS_WINDOWS = ($^O eq 'MSWin32'); sub new { my $class = shift; my %opts = (_send_via => 'http', _http_method => 'post', _wait => 1, @_); $opts{ua} = LWP::UserAgent->new; return bless \%opts, ref($class) || $class; } =head2 send_sms # Send an SMS message my $sent = $sender->send_sms( to => '447712345678', # phone number to which to send the message text => "Hello, world!", # content of the message ); =over =item _to The required C<_to> param should contain the phone number for the recipient. See the SMS::Send documentation for acceptable formats. =item _text The required C<_text> param should contain the text content that you wish to send. Normally this is limited to 160 characters though that restriction is NOT enforced by this module in order to take advantage of multiple-part messages ability available on some phones. Any newlines or carriage returns present are converted to spaces. =item _url The C<_url> param is optional and can be used to provide a callback URL specific to this message. See above for a complete description. =item _from The C<_from> param is optional and can be used to provide an email from address specific to this message. See above for a complete description. =item _from_id The C<_from_id> param is optional and can be used to provide a mesage originator specific to this message. See above for a complete description. =item _route The C<_route> param is optional and can be used to provide a reverse billing shortcode specific to this message. See above for a complete description. =item _wait The C<_wait> param is optional and can be used to suppress the automatic delivery confirmation stage on a per-message basis. See above for a complete description. =back =cut sub send_sms { my $self = shift; my %opts = (to => undef, text => undef, @_); if (not (defined($opts{to}) and defined($opts{text}))) { Carp::croak "You must provide a 'to' number and 'text' content"; } $opts{to} =~ s/^\+//s; $opts{text} =~ tr/\r\n/ /s; if (substr($self->{_send_via},0,4) eq 'http') { if (not (defined($self->{_login} || $self->{_user}) and defined($self->{_password}))) { Carp::croak "To send messages using http/s you must provide a Kapow username and password"; } my $protocol = $self->{_send_via} || 'http'; my $request_method = $self->{_http_method} || 'post'; my $response = do { my $r = undef; my %params = (username => $self->{_login} || $self->{_user} || "", password => $self->{_password} || "", mobile => $opts{to} || "", sms => $opts{text} || "", from_id => $opts{_from_id} || $self->{_from_id} || undef, route => $opts{_route} || $self->{_route} || undef, url => $opts{_url} || $self->{_url} || undef, returnid => 'TRUE', # enable message delivery tracking ); if ($request_method eq 'get') { $r = $self->{ua}->get(sprintf("%s://www.kapow.co.uk/scripts/sendsms.php?%s", $protocol, join('&', map {sprintf("%s=%s", $_, uri_escape($params{$_}))} grep {defined($params{$_})} keys %params))); } elsif ($request_method eq 'post') { $r = $self->{ua}->post("$protocol://www.kapow.co.uk/scripts/sendsms.php", {map {$_ => $params{$_}} grep {defined $params{$_}} keys %params}); } $r; }; if (defined($response) and $response->is_success) { my $reply = $response->content; $self->{_raw_response} = $reply; my ($word,$num_credits,$unique_id) = split /\s+/, $reply; if ($word eq 'OK') # message accepted { $self->{_new_credits_balance} = $num_credits; $self->{_msg_delivery_id} = $unique_id; $self->{_sent_at} = time(); # check delivery status, unless disabled my $check_delivery_status = 1; # default true if (defined($opts{_wait}) and ! $opts{_wait}) { undef $check_delivery_status } elsif (defined($self->{_wait}) and ! $self->{_wait}) { undef $check_delivery_status } if ($check_delivery_status) { wait_for_success: for (1..10) { my $reply = $self->delivery_status; if ($reply and substr($reply,0,1) ne 'N') { $self->{_delivered_at} = time(); last wait_for_success; } sleep 1 if $_ < 10; # wait and try again } } return $unique_id; } else # request went through but message not accepted { Carp::carp "SMS message not sent -- Kapow returned '$word'"; return; } } elsif (defined($response)) # failed lwp request, fatal { Carp::croak "Failed to issue HTTP request: " . $response->status_line; } else # perhaps a strange request method. we probably never made any request { Carp::croak "No HTTP request issued -- please ensure '_http_method' is set to either 'get' or 'post'"; } } elsif ($self->{_send_via} eq 'email') { my $send_method = $self->{_email_via} || ($IS_WINDOWS ? 'smtp' : 'sendmail'); my $sms = $opts{text} || ""; my $mobile = $opts{to} || ""; my %options = (To => "$opts{to}\@kapow.co.uk", Subject => $sms, Type => 'TEXT', Data => "", ); if (my $username = $self->{_login} || $self->{_user} and my $password = $self->{_password}) { $options{Data} = "$username\n$password\n"; } if (my $from = $opts{_from} || $self->{_from}) { $options{From} = $from; } my $message_as_string = ""; eval { my $msg = MIME::Lite->new(%options); $message_as_string = $msg->as_string; $msg->send($send_method); }; if ($@) { Carp::carp $@; return; } else { $self->{_sent_at} = time(); return $message_as_string } # send and forget } return; } =head2 delivery_status # What's the delivery status of the last message we sent? (available for http & https methods) my $status = $sender->delivery_status; # What's the delivery status for an arbitrary message we sent in the past? (pass it the return value of the send_sms method) my $status = $sender->delivery_status($sent); For messages sent via http/s you can check the delivery status of the message by calling this method. If called with no parameters then the most recent message sent out is checked. You can also provide the return value of the send_sms method as a parameter to check the delivery status for other messages. The module will use the same http or https setting that the sender object used to send the message (i.e. the _send_via param) as well as the same get/post setting. Messages sent by email cannot be checked for their delivery status. =cut sub delivery_status { my $self = shift; my $unique_id = shift || $self->{_msg_delivery_id} or Carp::croak "No message available for checking delivery_status"; if (not defined $self->{ua}) { Carp::croak "No user-agent available for checking delivery_status"; } my $protocol = $self->{_send_via} || 'http'; my $get_or_post = $self->{_http_method} || 'get'; my $response = undef; if ($get_or_post eq 'get') { $response = $self->{ua}->get(sprintf("%s://www.kapow.co.uk/scripts/chk_status.php?username=%s&returnid=%s", $protocol, uri_escape($self->{_login} || $self->{_user}), uri_escape($unique_id))); } elsif ($get_or_post eq 'post') { $response = $self->{ua}->post("$protocol://www.kapow.co.uk/scripts/chk_status.php", { username => $self->{_login} || $self->{_user}, returnid => $unique_id }); } else { Carp::croak "Unknown _http_method" } if (defined($response) and $response->is_success) { return $response->content; } return; } =head2 send_status The send_status method is an alias for delivery_status, provided for backward-compatibility. =cut sub send_status { my $self = shift; return $self->delivery_status(@_); } =head1 AUTHOR Jeremy Jones, C<< >> =head1 BUGS Please report any bugs or feature requests to C, or through the web interface at L. =head1 SUPPORT You can find documentation for this module with the perldoc command. perldoc SMS::Send::UK::Kapow =head1 ACKNOWLEDGEMENTS Adam Kennedy's SMS::Send module and Andrew Moore's SMS::Send::US::Ipipi module were useful for writing this one. =head1 COPYRIGHT & LICENSE Copyright 2009 Jeremy Jones, all rights reserved. This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. =cut 1; # End of SMS::Send::UK::Kapow