pluggit / http-client
A config-driven library to execute HTTP requests
Requires
- php: ^8.0
- guzzlehttp/psr7: ^1.0
- psr/log: ^2.0
Requires (Dev)
- behat/behat: ^3.1
- guzzlehttp/guzzle: ^6.0
- php-mock/php-mock: ^1.0
- phpspec/phpspec: ^7
- phpunit/php-code-coverage: ^2.2
- pimple/pimple: ^3.0
- pluggit/monitoring: ^4.0
- symfony/console: ^5.4
- symfony/var-dumper: ^5.4
- symfony/yaml: ^4.0
Suggests
- guzzlehttp/guzzle: Allows the library to send requests
- pimple/pimple: Allows to use the included service provider to register the client builder
- symfony/console: Allows to enable debug request executions in the console
- symfony/yaml: Allows to use yaml configuration files for the requests
- dev-master
- 3.0.0
- 2.1.1
- 2.1.0
- 2.0.0
- 1.2.5
- 1.2.4
- 1.2.3
- 1.2.2
- 1.2.1
- 1.2.0
- 1.1.3
- 1.1.2
- 1.1.1
- 1.1.0
- 1.0.11
- 1.0.10
- 1.0.9
- 1.0.8
- 1.0.7
- 1.0.6
- 1.0.5
- 1.0.4
- 1.0.3
- 1.0.2
- 1.0.1
- 1.0.0
- dev-PHP8-compatibility
- dev-PHP8-compatibility-2
- dev-upgrade/php8_v2
- dev-upgrade/php8
- dev-rebrand
- dev-array-replace
- dev-fix-log-1.2.4
- dev-docs_update
- dev-feature/WHD-825
- dev-json
- dev-dev_json
This package is not auto-updated.
Last update: 2025-04-02 15:24:05 UTC
README
Small PSR7 compatible library to perform http requests based on pre-configured set of requests
TL;DR
# requests.yml my_cool_api: endpoint: http://my_cool_api.com/v2 requests: get_user: path: /user/${USER_ID} create_user: path: /user/${USER_ID} method: POST headers: Content-Type: application/json body: name: ${NAME} email: ${EMAIL}
$coolApi = ClientBuilder::create() ->withYamlConfig('requests.yml') ->build('my_cool_api'); $user = $client->json('get_user', ['user_id' => $userId]); echo "Hello ".$user->name;
Table of contents
Installation
Require the package as usual
composer require pluggit/http-client
To use Guzzle as a sender (the default option), you have to require it manually
composer require guzzlehttp/guzzle "^6.0"
Requirements
php >=5.5
guzzlehttp/psr7 ^1.0
psr/log ^1.0
Note: This package relies on the implementation of guzzle for the PSR-7: HTTP message interfaces
Additional requirements
Guzzle
The library allows to use different senders to execute requests; and adapter for guzzlehttp/guzzle
it's included in the library but you must require the dependency manually
composer require guzzlehttp/guzzle "^6.0"
Optional requirements
Yaml parsing
If you want to use a Yaml file as a configuration source you can:
a) Use you own parser and pass the config as an array
b) Require symfony/yaml
as dependency and pass the yaml filepath to the builder
composer require symfony/yaml "^3.1"
Logging
Do you want to know why your request is failing? No problem, the library accepts any Psr-3 logger logger implementation (like Monolog) to write debug and error messages.
For a quick start you can use Symfony's console package to have debug messages in the console
composer require symfony/console "^3.1"
You will now be able to activate the console debug mode
$builder->withConsoleDebug();
Pimple
If you use pimple for building your dependencies you can use the provider Cmp\Http\Provider\HttpClientServiceProvider
to register the builder and the multi client.
composer require pimple/pimple "^3.0"
NOTE: See the annex for how to register the provider
Compatibility
This library has been tested with the following PHP versions and setups
PHP | guzzlehttp/psr7 | guzzlehttp/guzzle | result |
---|---|---|---|
5.5.25 | 1.3.1 | 6.2.2 | ✅ |
5.6.16 | 1.3.1 | 6.2.2 | ✅ |
7.0.7 | 1.3.1 | 6.2.2 | ✅ |
hhvm | 1.3.1 | 6.2.2 | ✅ |
Usage
The library allows to build requests from a set configuration values, execute them trough the client and get a response
Defining a configuration
The configuration is what tells the client how to build the requests, let's see some examples of configuration
my_users_api: endpoint: https://api.mysite.com headers: api_version: 1.2 options: timeout: 2 requests: list_users: path: /users options: timeout: 15 get_user: path: /users/${USER_ID} create_user: path: /users/${USER_ID} headers: - Content-Type: application/json method: POST # Another service the_comments_api: endpoint: http://messaging_service.com/v2/ ...
The first line defines a name for a service, in this case my_users_api
NOTE: You can see a full-fledged sample configuration file in config-sample/requests.yaml
for working with a REST API
Service configuration values
endpoint
(required): This is the endpoint for the all the request for this serviceheaders
: A key-value array with the headers to add to all the requests for this servicequery
: A key-value array with the query parameters to append in the URI of all the requests for this servicebody
: A key-value array with the post parameters to send in the body of all the requests for this serviceversion
: HTTP protocol version, default one1.1
options
: A key-value array with options to pass to the http client sender to modify it's behaviourrequests
: A key-value array with the allowed requests for this service
Request configuration values
The request configuration allows the set almost the same options as the service configuration, overwriting the values provided for the service, this allows to define a general behaviour for a service but tweak some options for specific requests, like for example, allowing a longer timeout
path
(required): Defines the path that follows the endpointmethod
: Defines the HTTP method for the request.GET
by defaultheaders
: A key-value array with the headers to add to this requestquery
: A key-value array with the query parameters to append in the URI of this requestbody
: A key-value array with the post parameters to send in the body of this requestversion
: HTTP protocol version, default one1.1
options
: A key-value array with options to pass to the http client sender to modify it's behaviour
Underlying client options
The request options allows to customize the client behaviour without the library having to know specific details about how to handle them.
Taking the provided integration with Guzzle as examples, we could have:
- Basic auth:
auth: ['myUser', 'myPass']
- Timeout:
timeout: 2
- Connect timeout:
timeout: 15
- Add a certificate:
['cert' => ['/path/server.pem', 'password']]
Configuration placeholders
The library allows to indicate the presence of dynamic values in this configuration parameters as placeholders
Placeholder rules
- Placeholders are allowed in path, query, headers and body only
- Use this format to add a placeholder:
${PLACEHOLDER_NAME}
- Placeholders tag names must be uppercase (although lowercase keys can be used when replacing them)
- You can have the same placeholder multiple times across all options, it will be replaced in all of them.
Replacing a placeholder in the request
To replace the placeholders, pass the values as paramters when creating the request
secure_api: endpoint: https://topsecret.com headers: api_key: MyPersonalApiKey token: ${TOKEN} requests: get_user: path: /users/${USER_ID}
$request = $client->create('secure_api', 'get_user', [ 'token' => $oath->sign($secret), 'user_id' => $userId ]);
Configuring the builder
The builder (Cmp\Http\ClientBuilder
) is an object that will help you create you client, easing the customization process. This are the options available:
$builder = ClientBuilder::create() ->withConfig($requests) ->withGuzzleSender($myCustomGuzzleClient) ->withConsoleDebug();
Once you've configured the builder you're ready to create clients for your http services
Building the client
There are 2 different clients that allow to create and execute requests:
Cmp\Http\Client\MultiClientInterface
: This client has access to all servicesCmp\Http\Client\ServiceClientInterface
: This client can execute request from a single service only
NOTE: You should try to inject always a ServiceClientInterface
in your services, this will prevent sideeffects triggering requests from other services by mistake
The available methods in the clients are:
create
: Creates a request (Cmp\Http\Message\Request
)send
: Sends a request returning aCmp\Http\Message\Response
They also provide some shortcuts to ease the use:
execute
: Creates and sends a request in a single calljson
: Creates and sends a request parsing a json responsejsonAsArray
: Creates and sends a request parsing as an array from a json response
Building a multi client
The easiest way to built the client is to use the method build
on the builder
$client = $builder->build();
Building a service client
To build a service client specify the name of the service when building the client in the last step of the builder
$myServiceApi = $builder->build('my_service');
Customizing the builds
The client require 3 dependencies to work, the builder has methods to override all of them with custom implementations
Sender
This is the class that sends the requests, it has to implement the Cmp\Http\Sender\SenderInterface
and return a Psr-7 compatible response
The builder will try to use the provided Cmp\Http\Sender\GuzzleSender
by default, to specify a different one use:
$builder->withSender($sender);
Even if you want to use guzzle, ou can also customize the internal client used
// tweak the client $customGuzzleClient = new Guzzlehttp\Guzzle(); // Pass it to the builder $builder->withGuzzleSender($customGuzzleClient);
Config
This is the only required dependency, a configuration defining the available the requests
$builder->withConfig($config);
If symfony/yaml
is installed, you can pass a filepath with a yaml configuration
$builder->withYamlConfig($yamlFile);
Logger
You can pass a Psr\Log\LoggerInterface
to receive debug and error messages
$builder->withLogger($logger);
If you want to debug the raw http request/responses in the console you can activate the debug output in the builder (You'll need to have symfony/console
installed)
$builder->withConsoleDebug();
Creating a request
To create a request you need to identify the service and the request that you want. Additionally you can pass dynamic values to substitute placeholders
$request = $client->request('weather', 'forecast', ['city' => $city]); // Service specific clients do not need the service name $request = $weather->request('forecast', ['city' => $city]);
The requests created are Cmp\Http\Message\Request
intances, implementing Psr\Http\Message\RequestInterface
, making them suitable to share between libraries and frameworks
The library request object provides some helpers to work with them in an easier way:
withQueryParameter($key, $value)
: Allows you to add or modify a query parameterwithPost(array $params)
: Allows to pass a key-value array of params to send as bodywithJsonPost(array $params)
: Allows to pass a key-value array of params; it will be codified as json and, the headerContent-Type: application/json
will be added__toString()
: This makes the request embeddable in string
Customising the request creation
If you provide a request factory to the builder that complais with Cmp\Http\RequestFactoryInterface
, you'll be able to extend the Request class and change the behaviour.
You may ask "hey, why would I like to do this for?"
For example to:
- Change all outgoing endpoints to a dummy server to prevent to executed certain requests on test environments
- Apply a configuration to all requests from all services
$factory = new TestEnvironmentsRequestFactory($config); $builder->withRequestFactory($factory);
Sending the request
After sending the request you'll receive a Cmp\Http\Message\Response
instance compatible with Psr\Http\Message\ResponseInterface
This response object also provides some helper methods:
json($asArray = true)
: Parses the body as a json an returns either astdClass
object or anarray
jsonAsArray()
: Same as before, but forcing the return type to be an array__toString()
: This makes the response embeddable in string
Exceptions
Not everything works as expected; to make your life easier handling this, the library provides a small set of exceptions:
Cmp\Http\Exception\RuntimeException
: This is the base exception used in the library, all the exceptions thrown within the library code uses or extends this oneCmp\Http\Exception\RequestBuildException
: This exception is thrown when the process of building a request cannot be completedCmp\Http\Exception\RequestExecutionException
: This exception is thrown when some error happens sending the request
REST API integration example
I'm going to show you how to perform the most common operation in a REST API.
Lets imagine that we want to interact with an API endpoint to manage our application users.
The user entity looks like:
{ "id": "1", "first_name": "John", "last_name": "Doe" }
Now let's define the typical requests in our configuration file
# Service definition my_app: # The endpoint is required endpoint: https://api.myapp.com/v1 options: auth: ['apikey', 'secret'] # At least one request is required too requests: list_users: path: /users get_user: path: /users/${USER_ID} delete_user: path: /users/${USER_ID} method: DELETE put_user: path: /users/${USER_ID} method: PUT headers: Content-Type: application/json body: first_name: ${FIRST_NAME} last_name: ${LAST_NAME} create_user: path: /users method: POST headers: Content-Type: application/json body: first_name: ${FIRST_NAME} last_name: ${LAST_NAME} update_user: path: /users/${USER_ID} method: PATCH headers: Content-Type: application/json # body: No need to define all the body params here, we can do it at request time
Building the API client
Let's build a service client for our api, as per the configuration, all the requests will include a basic authentication
$api = $builder->build('my_app');
###List users This call would return an array of users
// We are going to modify the request to pass a limit on the number of users to retrieve $users = $api->send($api->request('list_users')->withQueryParameter('limit', 25))->json();
Get a user
This call will inject the user id in the path of the uri to request the user
// The json shortcut creates, executes and parses the body response all in a single call $users = $api->json('get_user', ['user_id => 1]);
Create a user
Here we're going to replace the placeholders in the body array, if the api only accepts a json body, we have to add the correct content type header, like in the example so the request is created correctly
$user = $api->json('create_user', ['name' => 'Jane', 'last_name' => 'Roe']);
Replace a user
We can replace placeholders in both path or body params at the same time
$user = $api->json('put_user', ['user_id' => 1, 'name' => 'Jane', 'last_name' => 'Roe']);
Update a user
When we update some fields only, we don't known in advance what fields we are going to send, so it's better to decide later
$request = $api->create('update_user', ['user_id' => 1]); // Remember to assign the result, as the request return always a new instance when modifying it $request = $request->withPost(['name' => 'Jane']); $updatedUser = $api->send($request)->json();
Delete a user
Deleting a user is even easier
$api->execute('delete_user', ['user_id' => 1]);
Annexes
Pimple Service Provider
The library includes a service provider for Pimple to register a client in an easy way:
Cmp\Http\Provider\HttpClientServiceProvider
The provider will register the builder Cmp\Http\ClientBuilder
object at key http_client.builder
and a general purpouse multi-client Cmp\Http\Client\MultiClient
in http_client.client
, but it also accepts alias for both services
$container->register(new HttpClientServiceProvider('requester', 'client'), [ 'http_client.yaml' => 'config/requests.yml', ]); $builder = $container['requester']; // or http_client.builder $client = $container['client']; // or http_client.client
The options that the provider accepts are:
http_client.yaml
: A filepath with a ymal configuration filehttp_client.config
: A configuration array with the requestshttp_client.logger
: Pass a Psr\Logger\LoggerInterface object to add logging to the clienthttp_client.sender
: A custom implementation of theCmp\Http\Sender\SenderInterface
interfacehttp_client.guzzle
: A customGuzzleHttp\ClientInterface
object (useful to keep a reference)http_client.factory
: A customCmp\Http\RequestFactoryInterface
implementationhttp_client.debug
: Pass true and this will activate console output for debugging
Development environment
To build the test environment you'll need docker and docker-compose installed:
make dev
Running the tests
make unit make integration
You can run the tests only for a php version like this
make unit PHP_VERSION=5.6 make integration PHP_VERSION=5.6
Code-coverage
You can build a report for code coverage in HTML format. It will be available in bin/code-coverage
make code-coverage
Stop the environment
make nodev
Delete the environment
You can delete the docker images for a total clean-up
make nodev IMAGES=true