NOTE: We now require a psr/cache-implementation
so that the engine (and WSDL parsing) can be cached. Some examples are: symfony/cache
or cache/*-adapter
.
If you don't have a cache implementation installed, you can for example run:
composer require symfony/cache
Next, you can upgrade the soap-client:
composer require 'phpro/soap-client:^4.0.0' --update-with-dependencies
In V4, the engine factory changed again. This will be the final breaking change for the engine factory since we now have a generic way to create it for both runtime and code generation:
Change the engine inside your code generation configuration:
use Phpro\SoapClient\CodeGenerator\Config\Config;
use Phpro\SoapClient\Soap\CodeGeneratorEngineFactory;
use Phpro\SoapClient\Soap\DefaultEngineFactory;
use Phpro\SoapClient\Soap\EngineOptions;
use Soap\Wsdl\Loader\FlatteningLoader;
use Soap\Wsdl\Loader\StreamWrapperLoader;
return Config::create()
->setEngine($engine = DefaultEngineFactory::create(
EngineOptions::defaults('your.wsdl')
->withWsdlLoader(new FlatteningLoader(new StreamWrapperLoader())) // Or a PSR18-based loader ... :))
))
Note: In case you are using a custom configuration class by implementing the ConfigInterface
: this is not supported anymore.
Since code generation gets complexer and complexer, we decided to make the configuration more strict.
Regenerate classes:
./vendor/bin/soap-client generate:client --config=config/soap-client.php
./vendor/bin/soap-client generate:classmap --config=config/soap-client.php
./vendor/bin/soap-client generate:types --config=config/soap-client.php
./vendor/bin/soap-client generate:clientfactory --config=config/soap-client.php
Note: The generated code might have slightly changed for your project. Validate if you are still using the correct methods in your implementation.
Change the engine inside your (generated) ClientFactory:
$engine = DefaultEngineFactory::create(
EngineOptions::defaults($wsdl)
->withEncoderRegistry(
EncoderRegistry::default()
->addClassMapCollection(CalcClassmap::types())
->addBackedEnumClassMapCollection(CalcClassmap::enums())
)
// If you want to enable WSDL caching:
// ->withCache($yourPsr6CachePool)
// If you want to use Alternate HTTP settings:
// ->withWsdlLoader()
// ->withTransport()
// If you want specific SOAP setting:
// ->withWsdlParserContext()
// ->withWsdlServiceSelectionCriteria()
);
More information about the engine options can be found here.
In case you are using a non-default metadata strategy, you can now configure it in the code generation configuration:
use Phpro\SoapClient\CodeGenerator\Config\Config;
use Phpro\SoapClient\Soap\Metadata\Manipulators\DuplicateTypes\RemoveDuplicateTypesStrategy;
return Config::create()
->setDuplicateTypeIntersectStrategy(
new RemoveDuplicateTypesStrategy()
)
More information about the metadata options can be found here.
composer require 'phpro/soap-client:^3.0.0' --update-with-dependencies
Upgrading is a matter of changing the engine for code generation in the code-generation configuration file:
use Phpro\SoapClient\Soap\CodeGeneratorEngineFactory;
use Soap\Wsdl\Loader\FlatteningLoader;
use Soap\Wsdl\Loader\StreamWrapperLoader;
return Config::create()
->setEngine($engine = CodeGeneratorEngineFactory::create(
'your.wsdl',
new FlatteningLoader(new StreamWrapperLoader()) // Or a PSR18-based loader ... :)
))
Note: You can still use the default engine, yet you won't get the information for the enhanced type generation.
Regenerate classes:
./vendor/bin/soap-client generate:client --config=config/soap-client.php
./vendor/bin/soap-client generate:classmap --config=config/soap-client.php
./vendor/bin/soap-client generate:types --config=config/soap-client.php
V1 has been around for quite some time. Since the low-level SOAP parts became very stable and trustworthy, We decided to move them to a separate organisation. This will make it easier to make changes in both this soap-client and the more low-level SOAP parts. An additional benefit is that the robust SOAP fundamentals can be used by other SOAP clients as well. We hope this will lead to an even more stable SOAP core.
The focus of this package shifted to the generation of SOAP client code. This is the part that makes it easy to interact with your SOAP service. We decided to make it more opinionated, so that you have to deal with less strange SOAP related bugs than before.
By moving the low-level SOAP stuff out, we can now focus on improving WSDL metadata collection, which will results in more strictly typed generated code. You can expect this to come in one of the next releases.
If your application does not contain a PSR-18 client, you'll have to choose which HTTP client you want to use. This package expects some PSR implementations to be present in order to be installed:
- PSR-7:
psr/http-message-implementation
likenyholm/psr7
orguzzlehttp/psr7
- PSR-17:
psr/http-factory-implementation
likenyholm/psr7
orguzzlehttp/psr7
- PSR-18:
psr/http-client-implementation
likesymfony/http-client
orguzzlehttp/guzzle
Example implementations:
composer require symfony/http-client nyholm/psr7
composer require phpro/soap-client:^2.0
We suggest you to use the soap-client
CLI tools again if you want to upgrade and existing application:
./vendor/bin/soap-client wizard
From this point on, you can re-add the custom parts of your existing SOAP client back into the new SOAP client.
You can check the updated documentation in order to discover how you need to do specific actions like adding middleware in v2.
This update comes with some breaking changes:
The new php-soap organisation provides most of the interfaces that were previously in this package.
https://github.com/php-soap/engine
Provides following interfaces:
- Engine : Replaces the old
EngineInterface
- Metadata : Replaces the old
MetadataInterface
- MetadataProvider : Replaces the old
MetadataProviderInterface
- Driver : Replaces the old
DriverInterface
- Decoder : Replaces the old
DecoderInterface
- Encoder : Replaces the old
EncoderInterface
If you use any of these interfaces in your own code, you will need to replace them with the php-soap
alternatives.
Note:
- The metadata has slightly been changed : property and parameter collections are replacing plain arrays.
https://github.com/php-soap/psr18-transport/
- Transport : Replaces the old
HandlerInterface
By default we will be using a PSR-18 based transport.
The ExtSoapClientTransport
is still available, but won't be used by default anymore,
because it has many known issues in PHP's bug tracker.
The downside is that you have to decide which PSR-18 HTTP client you want to use before installing this package. For example:
composer require symfony/http-client nyholm/psr7
The HTTP middleware are now regular httplug plugins, which are also moved to another package: You can find a full list of plugins that can be used here:
https://github.com/php-soap/ext-soap-engine/
All ext-soap
quirks are now maintained in a separate package.
This will most likely have impact on:
- SoapOptions: The same as before, but without metadata manipulators. You can now configure them in the Engine factory.
- ClassMap: It now takes variadic
ClassMap
arguments as arguments instead of an array. - TypeConverter: Moved
- WsdlProvider: New WSDL providers were added to the new repo
If you have custom classes implementing one of the items above, they will require some love whilst upgrading.
Note:
- By default, the Transport does not contain any last request information anymore. There is a
TraceableTransport
available that you can decorate another transport with in order to get this functionality back.
- Symfony to LTS (4.4)
- PHP (^8.0)
- Removed a lot of old dependencies and suggestions
- Removed ext-*, since they are required in the specific php-soap packages
The custom event dispatchers are removed. We now support any PSR-14 event dispatcher.
Fully qualified class names will be used as event names. The old deprecated event names are now removed from the codebase.
The events won't contain the SOAP client anymore. We don't want them to be service containers. So instead, if you require the SOAP client in the event listeners, you need to inject them manually.
We don't work with a base client anymore.
Instead, a Caller
is injected into the client you fully own.
The caller is responsible for transporting the request.
We provide an engine caller and an event dispatching caller
so that your client keeps on working how you expect it to.
Example of generated client:
use Calculator\Type\Add;
use Calculator\Type\AddResponse;
use Phpro\SoapClient\Caller\Caller;
class CalculatorClient
{
/**
* @var Caller
*/
private $caller;
public function __construct(\Phpro\SoapClient\Caller\Caller $caller)
{
$this->caller = $caller;
}
public function add(Add $parameters) : AddResponse
{
return ($this->caller)('Add', $parameters);
}
}
We removed the debugging method from the soap-client.
Instead, you can either debug the request or result directly.
If you want to have access to the HTTP SOAP payload, we suggest adding a logger plugin to the HTTP client.
There is also a TraceableTransport
available that can be used to detect the last SOAP request and response like you would in the old client.
This implies that changes where done to:
- the client generator
- the client factory generator
- the configuration generator
In order to make the new Caller
system available for your SOAP client,
the factory now injects the caller into your SOAP client.
use Symfony\Component\EventDispatcher\EventDispatcher;
use Phpro\SoapClient\Soap\DefaultEngineFactory;
use Soap\ExtSoapEngine\ExtSoapOptions;
use Phpro\SoapClient\Caller\EventDispatchingCaller;
use Phpro\SoapClient\Caller\EngineCaller;
class CalculatorClientFactory
{
public static function factory(string $wsdl) : CalculatorClient
{
$engine = DefaultEngineFactory::create(
ExtSoapOptions::defaults($wsdl, [])
->withClassMap(CalculatorClassmap::getCollection())
);
$eventDispatcher = new EventDispatcher();
$caller = new EventDispatchingCaller(new EngineCaller($engine), $eventDispatcher);
return new CalculatorClient($caller);
}
}
You can opt-out on the event dispatching logic or decorate your own caller.
The DefaultEngineFactory
can now be configured with a transport and the metadata options.
Full example on how you can personalize your factory class:
use Http\Client\Common\PluginClient;
use Http\Discovery\Psr18ClientDiscovery;
use Phpro\SoapClient\Soap\DefaultEngineFactory;
use Phpro\SoapClient\Soap\Metadata\Manipulators\DuplicateTypes\RemoveDuplicateTypesStrategy;
use Phpro\SoapClient\Soap\Metadata\Manipulators\TypesManipulatorChain;
use Phpro\SoapClient\Soap\Metadata\MetadataOptions;
use Soap\ExtSoapEngine\ExtSoapOptions;
use Soap\ExtSoapEngine\Wsdl\Naming\Md5Strategy;
use Soap\ExtSoapEngine\Wsdl\PermanentWsdlLoaderProvider;
use Soap\Psr18Transport\Psr18Transport;
use Soap\Psr18Transport\Wsdl\Psr18Loader;
$httpClient = Psr18ClientDiscovery::find();
$engine = DefaultEngineFactory::create(
ExtSoapOptions::defaults($wsdl, [])
->withClassMap(CalculatorClassmap::getCollection())
->withWsdlProvider(
new PermanentWsdlLoaderProvider(
Psr18Loader::createForClient($httpClient),
new Md5Strategy(),
'target/location'
)
),
Psr18Transport::createForClient(
new PluginClient(
$httpClient,
[
$middleware1,
$middleware2,
]
)
),
MetadataOptions::empty()->withTypesManipulator(
new TypesManipulatorChain(
new RemoveDuplicateTypesStrategy()
)
)
);