Skip to content

Commit

Permalink
Detached from existing use case
Browse files Browse the repository at this point in the history
  • Loading branch information
Pilskalns committed Oct 29, 2017
1 parent cf207b4 commit c96762a
Show file tree
Hide file tree
Showing 2 changed files with 182 additions and 21 deletions.
136 changes: 136 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,136 @@
# F3-wcurl
F3 Wrapper for cURL to talk with external REST API. Handles authentication and response caching.

##### Preface

Each time I ask my browser to open Email, it jumps few redirects and opens main page. If my login have expired, it quickly asks for password and then continues previous task. Few hours later if same URL opened, browser still remembers login state and I directly get to my email.

Underneath so many actions, requests, but from user very simple action.

Same way I want my code to interact with API's. Here you have remote site, URL's for basic actions and at some point later now "Get enabled servers in this organization" or "Add user to this mailing list". By the way, if remote asks for authentication - here's you can tell it back and then try again.

### Configuration
Acess the class for first time
```
$wcurl = \wcurl::instance();
// or with optional parameters, which can also be set later
$wcurl = \wcurl::instance($remote_api_root,"yourClass::cb_do_login", $ttl);
```

#### Set remote API root
```
$wcurl->setRoot( "https://remote.site/api-root/v1/" );
```

#### Set login callback function
Function should be valid callback for `call_user_func()`

```
$wcurl->setLogin( 'yourClass::cb_do_login' );
```
If any request results in HTTP 401 or 403 code, `wcurl` calls Login callback function and then repeats original request. If again there is error, it is returned to original function result. `wcurl` stores cookies in temporary file unique to API root. This cookie file is included in every request.

Login function **example**
```
static function cb_do_login(){
$wcurl = \wcurl::instance();
$login = $wcurl->post("/login", array(
'login'=> 'my_user',
'password'=> 'covfefe' )
);
}
```


#### Set default response cache expiration time

Integer, in seconds

```
$wcurl->setTTL( 3600 );
```

### How to use it

#### GET

```
$response = $wcurl->get( $url );
```

#### POST

```
$response = $wcurl->post( $url, $body );
```
If body is array, it will be passed trough `json_encode`. Possible improvement to allow HTTP form serialization from array.
#### Using named routes (rests)

When constructing long remote URL's, it's easier to remember them by short keywords, especially if they are called from multiple places. Like `allmembers` instead of `/lists/members/all/pages`.

Sometimes these contain also unique parameters, which needs to be filled in per each request.

To set named routes, pass simple key => value array. You can call this multiple times, with array being merged to already existing routes. I.e. from `.ini` configuration
```
$wcurl->setRests( array(
'allmembers' => '/lists/members/all/pages',
'withVariable' => '/lists/members/%%memberID%%/pages',
'updateEmail' => '/lists/members/%%memberID%%/update'
) );
```
URL variables are wrapped in two `%` from both sides.

To use named route, pass it's name instead of full URL.

```
$response = $wcurl->get( 'allmembers' );
```

### Fill / URLs with variables variables

To fill existing URL either from your custom or trough `setRests()` available ones, pass `$key => $value` array as last parameter to method.

```
$response = $wcurl->get( 'withVariable', array('memberID' => 'abcID') );
```
or
```
$wcurl->post( 'updateEmail',
array('email'=>'andzs@pilskalns.lv'),
array('memberID' =>'abcID')
);
```

### Using INI configuration

If you need only occasionally call external API's, adding configuration from `beforeRoute` will initialize class every time. PHP is fast, but still.

If you put all configuration in your main `ini` file, class can be initialized only on first required use of it. I.e. when your code decides to send get(). At that moment, if class is not registered in Prefab, it will be built from INI config exactly as needed.

```
[wcurl]
root=http://mysite.api/v1
ttl=3600
cb_login=yourClass::cb_do_login
[wcurl.rests]
allmembers=/lists/members/all/pages
withVariable=/lists/members/%%memberID%%/pages
```

### Using with multiple API's
When calling `\wcurl::instance()` it is built like singleton class, thus in any place in code, same instance is returned. To force fresh instance from class, use something like
```
$apiTwo = new wcurl([$root] [,$cb_login] [,$ttl]);
```
And there `$apiTwo` can be stored in F3 hive.


There's a lot to improve, currently this automatically support only 401/403 login callback, but could have optional hooks for every HTTP status.

Feedback, PR's and suggestions welcome.


Though this plugin could be developed solely as standalone class/tool, it will be used in context of [Fat Free Framework](https://fatfreeframework.com) (F3), so naturally got few cool F3 features as dependencies - `Prefab`, `Cache` and `Web`.

Thanks to F3 developers for this wonderful framework. If you are looking for such tool, but don't use F3, then think twice - Why you are not using F3 yet?
67 changes: 46 additions & 21 deletions lib/wcurl.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,47 +2,71 @@

class wcurl extends \Prefab {

private $cb_login, $rests;
private $cb_login, $root, $tll, $cookie;

private $ttl = 60;
private $rests = array();

function __construct($cb_login = null, $request_root = null, $ttl = 60) {
function __construct($root = null, $cb_login = null, $ttl = null) {
global $f3;
if(!strlen($f3->get('wcurl_cookie'))){
$f3->set('wcurl_cookie', $f3->get('TEMP').'wcurl_cookie.tmp' );
touch($f3->get('wcurl_cookie'));

$f3->exists('wcurl.root', $this->root);

if($root){
$this->root = $root;
}

self::setCookie();

$f3->exists('wcurl.cb_login', $this->cb_login);

if($cb_login){
$this->cb_login = $cb_login;
call_user_func($cb_login);
// call_user_func($cb_login);
}

if( !self::setTTL($ttl) && !self::setTTL($f3->get('wcurl.ttl')) ){
$this->ttl = 60;
}
if(is_array($f3->get('wcurl.rests')))
$f3->exists('wcurl.rests', $this->rests);


}

function __destruct(){
// if($this->ch){
// curl_close($this->ch);
// }
}

function setLogin($cb){
$this->cb_login = $cb;
}
function setIsLogged($cb){
$this->cb_islogged = $cb;
function setRoot($root){
$this->root = $root;
self::setCookie();
}
private function setCookie(){
global $f3;
$this->cookie = $f3->get('TEMP').'wcurl.cookie.'.
\Web::instance()->slug($this->root).
'.tmp';
touch($this->cookie);
}

function setRests($arr){
if(is_array($arr))
$this->rests = $arr;
$this->rests = array_merge($this->rests, $arr);
}
function setCachettl($int){
function getRests(){
return $this->rests;
}
function setTTL($int){
if ((is_int($int) || ctype_digit($int)) && (int)$int > 0 ) {
$this->ttl = $int;
return true;
}
return false;
}

function fillRESTS($url, $fill){
private function fillRESTS($url, $fill){
if(array_key_exists($url,$this->rests)){
$url = $this->rests[$url];
}
Expand Down Expand Up @@ -82,7 +106,9 @@ function get($url, $fill = null, $usecache = true){
$value['fromcache'] = false;
return $request;
}
function post($url, $body = null){
function post($url, $body = null, $fill = null){

$url = self::fillRESTS($url, $fill);

if(is_array($body)){
$body = json_encode($body);
Expand All @@ -96,7 +122,7 @@ function post($url, $body = null){
));
}

function curl_send($params = array(), $nested = false){
private function curl_send($params = array(), $nested = false){
global $f3;

$default = array(
Expand All @@ -105,11 +131,11 @@ function curl_send($params = array(), $nested = false){
CURLOPT_SSL_VERIFYPEER => true,
CURLOPT_SSL_VERIFYHOST => 2,
CURLOPT_FOLLOWLOCATION => true,
CURLOPT_COOKIEJAR => realpath($f3->get('wcurl_cookie')),
CURLOPT_COOKIEFILE => realpath($f3->get('wcurl_cookie')),
CURLOPT_COOKIEJAR => realpath($this->cookie),
CURLOPT_COOKIEFILE => realpath($this->cookie),
);

$url = trim($f3->get('nimbus.root'),'/').
$url = trim($this->root,'/').
'/'.trim($params[CURLOPT_URL],'/');

$setparams = $params;
Expand Down Expand Up @@ -153,7 +179,6 @@ function($curl, $header) use (&$headers){
break;
}
}

if($response){
$return = array('status' => $status,
'response' => $response,
Expand Down

0 comments on commit c96762a

Please sign in to comment.