czim / laravel-service
Basic webservice framework for Laravel.
Installs: 24 842
Dependents: 1
Suggesters: 0
Security: 0
Stars: 4
Watchers: 2
Forks: 3
Open Issues: 3
Requires
- php: ^8.1
- czim/laravel-dataobject: ^3.0
- guzzlehttp/guzzle: ^7.0
- illuminate/support: ^9
Requires (Dev)
- mockery/mockery: ^1.4
- nunomaduro/larastan: ^2.2
- orchestra/testbench: ^7.0
- phpstan/phpstan-mockery: ^1.1
- phpstan/phpstan-phpunit: ^1.1
- phpunit/phpunit: ^9.5
- dev-master
- 3.0.2
- 3.0.1
- 3.0.0
- 2.0.0
- 1.1.0
- 1.0.2
- 1.0.1
- 1.0.0
- 0.9.36
- 0.9.35
- 0.9.34
- 0.9.33
- 0.9.32
- 0.9.31
- 0.9.30
- 0.9.29
- 0.9.28
- 0.9.27
- 0.9.26
- 0.9.25
- 0.9.24
- 0.9.23
- 0.9.22
- 0.9.21
- 0.9.20
- 0.9.19
- 0.9.18
- 0.9.17
- 0.9.16
- 0.9.15
- 0.9.14
- 0.9.13
- 0.9.12
- 0.9.11
- 0.9.10
- 0.9.9
- 0.9.8
- 0.9.7
- 0.9.6
- 0.9.5
- 0.9.4
- 0.9.3
- 0.9.2
- 0.9.1
- 0.9.0
- dev-old-compatible
- dev-laravel-5.4
This package is auto-updated.
Last update: 2024-10-27 17:51:43 UTC
README
Basic framework for making standardized but flexible webservice classes.
Version Compatibility
Install
Via Composer
$ composer require czim/laravel-service
Unless you're using auto-discovery, add this line of code to the providers array
located in your config/app.php
file:
\Czim\Service\Providers\ServiceServiceProvider::class,
Requirements
To use the SshFileService
, you will require the libssh2
PHP extension.
Usage
Services
The basic approach to using a Service in three steps:
- Instantiate a Service
- passing in the defaults (optional)
- passing in an interpreter instance (optional)
- Perform a
call()
on the service instance
For a simple SOAP service, this might be as follows:
// Set up defaults $defaults = new \Czim\Service\Requests\ServiceSoapRequestDefaults(); $defaults->setLocation('http://www.webservicex.net/globalweather.asmx?WSDL') ->setOptions([ 'trace' => true, 'exceptions' => true, 'features' => SOAP_SINGLE_ELEMENT_ARRAYS, ]); // Instantiate service, with a to-array interpreter $service = new SoapService( $defaults, new \Czim\Service\Interpreters\BasicSoapXmlAsArrayInterpreter() ); // Prepare a specific request $request = new ServiceSoapRequest(); $request->setBody([ 'CityName' => 'Tokyo', 'CountryName' => 'Japan', ]); // Perform the call, which will return a ServiceReponse object $response = $service->call('GetWeather', $request);
Available Services
RestService
: for HTTP REST requests, requires aServiceRestRequest
object for defaults/requests.- See the section about RestServices below.
RestCurlService
: same as above, but uses cURL directly instead of guzzleSoapService
: for SOAP requests, requires aServiceSoapRequest
object for defaults/requests.- See the note about BeSimple below.
FileService
: for retrieving the contents of a single file, requires aServiceSshRequest
object for defaults/requests.MultiFileService
: for retrieving the contents of multiple files and merging their interpretations, requires aServiceSshRequest
object for defaults/requests.SshFileService
: like the MultiFileService, but retrieves files by logging into an SSH server. See requirements above.
See this list of examples for more information.
RestServices and HTTP methods
One thing to look out for is a potential confusion over the 'method' when using the RestService
.
The $method
parameter in the service's call()
does not refer to the HTTP Method (GET
, POST
, etc).
For cross-services compatibility purposes it is used to set the URI path instead, since that best corresponds to the 'method' for SOAP calls.
The HTTP Method is to be separately set on the service before making the call, with setHttpMethod()
.
So making a DELETE
call to /comments/13
may be set up as follows:
// Set up the default and base URI for the service $defaults = new \Czim\Services\Requests\ServiceRestRequestDefaults(); $defaults->setLocation('http://base.service.url.com/v1'); $defaults->setHttpMethod($service::METHOD_DELETE); // Instantiate a new service with the defaults $service = new \Czim\Service\Services\RestService($defaults); // Perform a call to delete comment 13 $result = $service->call('comments/13');
If not set, the default HTTP method is POST
.
Alternatively, the HTTP method may be configured per-call without affecting the default, by setting it in a ServiceRequest
parameter:
// Set up the default and base URI for the service $defaults = new \Czim\Services\Requests\ServiceRestRequestDefaults(); $defaults->setLocation('http://base.service.url.com/v1'); // Instantiate a new service with the defaults $service = new \Czim\Service\Services\RestService($defaults); // Create a request object with an HTTP method $request = new \Czim\Service\Requests\ServiceRestRequest(); $request->setHttpMethod($service::METHOD_DELETE); // Perform a call to delete comment 13 $result = $service->call('comments/13', $request);
The order of precedence is:
- HTTP method set in request for call
- HTTP method set in default request for service
- HTTP method defined in service directly
Interpreters
Services always return a ServiceResponse
for each call.
The content of the response (mainly the data accessible through getData()
, but also its other properties) is determined by the ServiceInterpreter
instance set for the service.
ServiceInterpreters are responsible for interpreting and converting the raw data retrieved to whatever (standardized) format and response content you prefer.
Standard available interpreters:
BasicDefaultInterpreter
: sets up response, but does not manipulate the raw data in any wayBasicJsonInterpreter
: converts a string with raw JSON to an object or associative arrayBasicQueryStringInterpreter
: converts a query string to an object or arrayBasicRawXmlInterpreter
: converts a string with raw XML to SimpleXML objectsBasicRawXmlAsArrayInterpreter
: same, but converts to an associative arrayBasicSoapXmlInterpreter
: makes response for SimpleXML object returned by a SoapClientBasicSoapXmlAsArrayInterpreter
: same, but converts to an associative array
It is recommended to extend these or roll your own interpreter for complex services.
I often build interpreters that use czim/laravel-dataobject
to convert raw data into validatable, specific data objects.
See the AbstractServiceInterpreter source for hints and useful methods.
Interpreter Decorators
Some decorators for interpreters are provided:
FixXmlNamespacesDecorator
: rewrites relative path namespaces to absolute in raw XML, since some XML may be uninterpretable otherwise (hacky).RemoveXmlNamespacesDecorator
: removes XML namespaces from raw XML entirely (hacky).
They follow the standard decorator pattern for interpreters, so they may be used as follows:
// Decorate a raw XML interpreter with a namespace removal decorator $interpreter = new \Czim\Service\Interpreters\Decorators\RemoveXmlNamespacesDecorator( new \Czim\Service\Interpreters\Decorators\BasicRawXmlInterpreter() ); // Inject the interpreter into a new service $service = new \Czim\Service\Services\FileService(null, $interpreter);
The AbstractValidationPreDecorator
may be extended to easily set up validation of raw data before the decorated interpreter does its business.
Likewise the AbstractValidationPostDecorator
may be extended to validate the ServiceResponse
object returned after the interpreter has treated the response.
Service Collection
By way of wrapping multiple services, this package offers a ServiceCollection
.
This is a simple extension of the Illuminate\Support\Collection
, which may only be used to store ServiceInterface
objects.
You can set one up just as you would a normal Collection:
$services = new \Czim\Service\Collections\ServiceCollection(); // Adding services $services->put('service name', $service); // Performing calls on a service $service->get('service name') ->call('someMethod', [ 'param' => 'value' ]);
Note that calling get()
on the collection for a non-existant service will throw an exception; as will trying to store something other than a ServiceInterface
in it.
Notes
BeSimple SoapClient
The BeSimpleSoapService
service class is set up to use the BeSimple SoapClient.
The only thing required to use it is to include the package:
composer require "besimple/soap-client"
DOMDocument parsing as an alternative to SimpleXml
When using the DomDocumentBasedXmlParser
, note that this will not return a SimpleXml-type object, but a DOMElement
.
To sensibly use this, convert it to an array using the DomObjectToArrayConverter
.
This means that when rebinding or injecting one of these, treat them as a pair.
Note that the DOMDocument method is slower and uses significantly more memory. If you have no specific reason to use this, stick to the default.
Contributing
Please see CONTRIBUTING for details.
Credits
License
The MIT License (MIT). Please see License File for more information.