Compare commits
42 Commits
499abe195e
..
master
| Author | SHA1 | Date | |
|---|---|---|---|
| 172128b37b | |||
| 8bb522ec33 | |||
| f32d3a4877 | |||
| 0fe449538e | |||
| c797cb9f24 | |||
| e884b0e97d | |||
| 9b153dacd8 | |||
| fdf7e89afe | |||
| 7baa903f7a | |||
| 3e0ba5f209 | |||
| a5598130db | |||
| e6ceea1b1b | |||
| 064e8a8054 | |||
| 52ecda573b | |||
| a88999cd18 | |||
| 40d396908d | |||
| f01509cab1 | |||
| 51941b88ee | |||
| 222194e554 | |||
| 78acb54845 | |||
| b4ccced253 | |||
| 2106f15d7f | |||
| 63b8173399 | |||
| df0f8f36f7 | |||
| a6d844bd01 | |||
| 3b054a640b | |||
| ab132b67e8 | |||
| fd19e442b6 | |||
| aae17f10a6 | |||
| e7a49138bb | |||
| 644f3697c7 | |||
| 31d385d3d4 | |||
| ebc7ff4f2f | |||
| 67a58a5277 | |||
| 0a7f86dc33 | |||
| 5d0c9c7276 | |||
| 1fa3fdddd0 | |||
| cbdc38e98e | |||
| 6137098872 | |||
| 1e9b99c7a6 | |||
| a65593b4f9 | |||
| e7d7d18f77 |
@@ -0,0 +1 @@
|
|||||||
|
node_modules
|
||||||
Executable
+12
@@ -0,0 +1,12 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
FILES=$(git diff --cached --name-only --diff-filter=ACMR -- '*.php' '*.html' '*.yaml' | sed 's| |\\ |g')
|
||||||
|
[ -z "$FILES" ] && exit 0
|
||||||
|
|
||||||
|
# Prettify all selected files
|
||||||
|
echo "$FILES" | xargs ./node_modules/.bin/prettier --ignore-unknown --write
|
||||||
|
|
||||||
|
# Add back the modified/prettified files to staging
|
||||||
|
echo "$FILES" | xargs git add
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
vendor
|
||||||
+11
@@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"printWidth": 120,
|
||||||
|
"tabWidth": 2,
|
||||||
|
"singleQuote": true,
|
||||||
|
|
||||||
|
"semi": true,
|
||||||
|
"trailingComma": "es5",
|
||||||
|
"arrowParens": "always",
|
||||||
|
|
||||||
|
"phpVersion": "7.1"
|
||||||
|
}
|
||||||
Vendored
+17
@@ -0,0 +1,17 @@
|
|||||||
|
{
|
||||||
|
"editor.formatOnSave": true,
|
||||||
|
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
||||||
|
"[php]":
|
||||||
|
{
|
||||||
|
"editor.formatOnSave": true,
|
||||||
|
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
||||||
|
},
|
||||||
|
"search.exclude": {
|
||||||
|
"**/vendor": true,
|
||||||
|
"**/node_modules": true,
|
||||||
|
"yarn.lock": true,
|
||||||
|
"package.json": true,
|
||||||
|
"tsconfig.json": true,
|
||||||
|
".*": true,
|
||||||
|
},
|
||||||
|
}
|
||||||
+37
-15
@@ -4,8 +4,11 @@ use System\Classes\PluginBase;
|
|||||||
|
|
||||||
use App;
|
use App;
|
||||||
use Config;
|
use Config;
|
||||||
|
use Illuminate\Database\DatabaseManager;
|
||||||
|
use Illuminate\Support\Facades\Broadcast;
|
||||||
use Illuminate\Foundation\AliasLoader;
|
use Illuminate\Foundation\AliasLoader;
|
||||||
use GermanAirlinesVa\Graphql\Classes\GraphqlServiceProvider;
|
use GermanAirlinesVa\Graphql\Classes\GraphqlServiceProvider;
|
||||||
|
use Nuwave\Lighthouse\Subscriptions\SubscriptionServiceProvider;
|
||||||
|
|
||||||
class Plugin extends PluginBase
|
class Plugin extends PluginBase
|
||||||
{
|
{
|
||||||
@@ -15,13 +18,39 @@ class Plugin extends PluginBase
|
|||||||
|
|
||||||
public function registerSettings()
|
public function registerSettings()
|
||||||
{
|
{
|
||||||
|
return [
|
||||||
|
'settings' => [
|
||||||
|
'label' => 'germanairlinesva.graphql::lang.settings.label',
|
||||||
|
'description' => 'germanairlinesva.graphql::lang.settings.description',
|
||||||
|
'category' => 'system::lang.system.categories.cms',
|
||||||
|
'icon' => 'icon-compress',
|
||||||
|
'class' => 'GermanAirlinesVa\Graphql\Models\Settings',
|
||||||
|
'order' => 1000,
|
||||||
|
'keywords' => 'graphql',
|
||||||
|
'permissions' => ['germanairlinesva.graphql.schemas'],
|
||||||
|
],
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function boot()
|
public function boot()
|
||||||
{
|
{
|
||||||
App::make('October\Rain\Support\ClassLoader')->addDirectories('graphql');
|
App::make('October\Rain\Support\ClassLoader')->addDirectories('graphql');
|
||||||
|
|
||||||
$this->bootPackages();
|
$this->bootPackages();
|
||||||
|
|
||||||
App::register(GraphqlServiceProvider::class);
|
App::register(GraphqlServiceProvider::class);
|
||||||
|
App::register(SubscriptionServiceProvider::class);
|
||||||
|
|
||||||
|
$this->app->singleton(DatabaseManager::class, function ($app) {
|
||||||
|
return $app->make('db');
|
||||||
|
});
|
||||||
|
Broadcast::routes([
|
||||||
|
'prefix' => '',
|
||||||
|
'middleware' => 'GermanAirlinesVA\\Graphql\\Classes\\Authentication',
|
||||||
|
]);
|
||||||
|
|
||||||
|
Config::set('database.connections.germanairlinesva_graphql', Config::get('germanairlinesva.graphql::connection'));
|
||||||
|
Config::push('system.unencrypt_cookies', 'graphql-session-id');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function bootPackages()
|
public function bootPackages()
|
||||||
@@ -36,29 +65,23 @@ class Plugin extends PluginBase
|
|||||||
$packages = Config::get($pluginNamespace . '::packages');
|
$packages = Config::get($pluginNamespace . '::packages');
|
||||||
|
|
||||||
// Boot each package
|
// Boot each package
|
||||||
foreach ($packages as $name => $options)
|
foreach ($packages as $name => $options) {
|
||||||
{
|
|
||||||
// Setup the configuration for the package, pulling from this plugin's config
|
// Setup the configuration for the package, pulling from this plugin's config
|
||||||
if (!empty($options['config']) && !empty($options['config_namespace']))
|
if (!empty($options['config']) && !empty($options['config_namespace'])) {
|
||||||
{
|
|
||||||
Config::set($options['config_namespace'], $options['config']);
|
Config::set($options['config_namespace'], $options['config']);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Register any Service Providers for the package
|
// Register any Service Providers for the package
|
||||||
if (!empty($options['providers']))
|
if (!empty($options['providers'])) {
|
||||||
{
|
foreach ($options['providers'] as $provider) {
|
||||||
foreach ($options['providers'] as $provider)
|
App::register($provider);
|
||||||
{
|
|
||||||
App::register($provider);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Register any Aliases for the package
|
// Register any Aliases for the package
|
||||||
if (!empty($options['aliases']))
|
if (!empty($options['aliases'])) {
|
||||||
{
|
foreach ($options['aliases'] as $alias => $path) {
|
||||||
foreach ($options['aliases'] as $alias => $path)
|
$aliasLoader->alias($alias, $path);
|
||||||
{
|
|
||||||
$aliasLoader->alias($alias, $path);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -66,4 +89,3 @@ class Plugin extends PluginBase
|
|||||||
return $packages;
|
return $packages;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
# Relations
|
||||||
|
|
||||||
|
## GraphQLKeys
|
||||||
|
|
||||||
|
- BelongsTo Member (External DB, creation in Social Plugin)
|
||||||
+61205
File diff suppressed because one or more lines are too long
@@ -0,0 +1,25 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\Graphql\Classes;
|
||||||
|
|
||||||
|
use Closure;
|
||||||
|
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
|
||||||
|
|
||||||
|
class Authentication
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Handle an incoming request.
|
||||||
|
*
|
||||||
|
* @param \Illuminate\Http\Request $request
|
||||||
|
* @param \Closure $next
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function handle($request, Closure $next)
|
||||||
|
{
|
||||||
|
if ($request->header('Gql-Session') === 'SUPER_SECRET_KEY_HEADER') {
|
||||||
|
return '';
|
||||||
|
} else {
|
||||||
|
throw new AccessDeniedHttpException();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $next($request);
|
||||||
|
}
|
||||||
|
}
|
||||||
+11
-13
@@ -6,19 +6,17 @@ use Illuminate\Http\Request;
|
|||||||
use Nuwave\Lighthouse\Support\Contracts\CreatesContext as LighthouseCreatesContext;
|
use Nuwave\Lighthouse\Support\Contracts\CreatesContext as LighthouseCreatesContext;
|
||||||
use Nuwave\Lighthouse\Support\Contracts\GraphQLContext;
|
use Nuwave\Lighthouse\Support\Contracts\GraphQLContext;
|
||||||
|
|
||||||
|
|
||||||
class CreatesContext implements LighthouseCreatesContext
|
class CreatesContext implements LighthouseCreatesContext
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
/**
|
* Generate GraphQL context.
|
||||||
* Generate GraphQL context.
|
*
|
||||||
*
|
* @param Request $request
|
||||||
* @param Request $request
|
*
|
||||||
*
|
* @return GraphQLContext
|
||||||
* @return GraphQLContext
|
*/
|
||||||
*/
|
public function generate(Request $request): GraphQLContext
|
||||||
public function generate(Request $request): GraphQLContext
|
{
|
||||||
{
|
return new SchemaContext($request);
|
||||||
return new SchemaContext($request);
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,74 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\Graphql\Classes;
|
||||||
|
|
||||||
|
use Illuminate\Support\Str;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use GraphQL\Type\Definition\ResolveInfo;
|
||||||
|
use Nuwave\Lighthouse\Subscriptions\Subscriber;
|
||||||
|
use Nuwave\Lighthouse\Schema\Types\GraphQLSubscription;
|
||||||
|
use Nuwave\Lighthouse\Support\Contracts\GraphQLContext;
|
||||||
|
|
||||||
|
class Dummy extends GraphQLSubscription
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Check if subscriber is allowed to listen to the subscription.
|
||||||
|
*
|
||||||
|
* @param \Nuwave\Lighthouse\Subscriptions\Subscriber $subscriber
|
||||||
|
* @param \Illuminate\Http\Request $request
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function authorize(Subscriber $subscriber, Request $request): bool
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filter which subscribers should receive the subscription.
|
||||||
|
*
|
||||||
|
* @param \Nuwave\Lighthouse\Subscriptions\Subscriber $subscriber
|
||||||
|
* @param mixed $root
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
public function filter(Subscriber $subscriber, $root): bool
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Encode topic name.
|
||||||
|
*
|
||||||
|
* @param \Nuwave\Lighthouse\Subscriptions\Subscriber $subscriber
|
||||||
|
* @param string $fieldName
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function encodeTopic(Subscriber $subscriber, string $fieldName): string
|
||||||
|
{
|
||||||
|
// Create a unique topic name based on the `author` argument
|
||||||
|
return Str::snake($fieldName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Decode topic name.
|
||||||
|
*
|
||||||
|
* @param string $fieldName
|
||||||
|
* @param \App\Post $root
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function decodeTopic(string $fieldName, $root): string
|
||||||
|
{
|
||||||
|
return Str::snake($fieldName);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolve the subscription.
|
||||||
|
*
|
||||||
|
* @param \App\Post $root
|
||||||
|
* @param array<string, mixed> $args
|
||||||
|
* @param \Nuwave\Lighthouse\Support\Contracts\GraphQLContext $context
|
||||||
|
* @param \GraphQL\Type\Definition\ResolveInfo $resolveInfo
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function resolve($root, array $args, GraphQLContext $context, ResolveInfo $resolveInfo): array
|
||||||
|
{
|
||||||
|
return [$root];
|
||||||
|
}
|
||||||
|
}
|
||||||
+23
-31
@@ -5,42 +5,34 @@ namespace GermanAirlinesVa\Graphql\Classes;
|
|||||||
use Lang;
|
use Lang;
|
||||||
use Cms\Classes\Page;
|
use Cms\Classes\Page;
|
||||||
|
|
||||||
|
|
||||||
class Graph extends Page
|
class Graph extends Page
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* @var string The container name associated with the model.
|
||||||
|
*/
|
||||||
|
protected $dirName = 'graphs';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var string The container name associated with the model.
|
* @var array The attributes that are mass assignable.
|
||||||
*/
|
*/
|
||||||
protected $dirName = 'graphs';
|
protected $fillable = ['title', 'description', 'markup', 'settings', 'code'];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var array The attributes that are mass assignable.
|
* @var array The rules to be applied to the data.
|
||||||
*/
|
*/
|
||||||
protected $fillable = [
|
public $rules = [
|
||||||
'title',
|
'title' => 'required',
|
||||||
'description',
|
];
|
||||||
'markup',
|
|
||||||
'settings',
|
|
||||||
'code'
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
public function getCodeClassParent(): string
|
||||||
* @var array The rules to be applied to the data.
|
{
|
||||||
*/
|
return GraphCode::class;
|
||||||
public $rules = [
|
}
|
||||||
'title' => 'required'
|
|
||||||
];
|
|
||||||
|
|
||||||
public function getCodeClassParent() : string {
|
public function runComponents()
|
||||||
return GraphCode::class;
|
{
|
||||||
|
foreach ($this->components as $component) {
|
||||||
|
$component->onRun();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
public function runComponents()
|
|
||||||
{
|
|
||||||
foreach ($this->components as $component) {
|
|
||||||
$component->onRun();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
+32
-34
@@ -8,48 +8,46 @@ use October\Rain\Extension\Extendable;
|
|||||||
*/
|
*/
|
||||||
class GraphCode extends CodeBase
|
class GraphCode extends CodeBase
|
||||||
{
|
{
|
||||||
|
public $graph;
|
||||||
|
|
||||||
public $graph;
|
/**
|
||||||
|
* Creates the object instance.
|
||||||
|
* @param \GermanAirlinesVa\Graphql\Classes\Graph $graph Specifies the Headstart graph.
|
||||||
|
* @param null
|
||||||
|
* @param \GermanAirlinesVa\Graphql\Classes\GraphController $controller Specifies the Graph controller.
|
||||||
|
*/
|
||||||
|
public function __construct($graph, $layout, $controller)
|
||||||
|
{
|
||||||
|
$this->graph = $graph;
|
||||||
|
$this->controller = $controller;
|
||||||
|
|
||||||
/**
|
Extendable::__construct();
|
||||||
* Creates the object instance.
|
}
|
||||||
* @param \GermanAirlinesVa\Graphql\Classes\Graph $graph Specifies the Headstart graph.
|
|
||||||
* @param null
|
|
||||||
* @param \GermanAirlinesVa\Graphql\Classes\GraphController $controller Specifies the Graph controller.
|
|
||||||
*/
|
|
||||||
public function __construct($graph, $layout, $controller)
|
|
||||||
{
|
|
||||||
$this->graph = $graph;
|
|
||||||
$this->controller = $controller;
|
|
||||||
|
|
||||||
Extendable::__construct();
|
public function __get($name)
|
||||||
|
{
|
||||||
|
if (isset($this->graph->components[$name]) || isset($this->layout->components[$name])) {
|
||||||
|
return $this[$name];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __get($name)
|
if (($value = $this->graph->{$name}) !== null) {
|
||||||
{
|
return $value;
|
||||||
if (isset($this->graph->components[$name]) || isset($this->layout->components[$name])) {
|
|
||||||
return $this[$name];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (($value = $this->graph->{$name}) !== null) {
|
|
||||||
return $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (array_key_exists($name, $this->controller->vars)) {
|
|
||||||
return $this[$name];
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __set($name, $value)
|
if (array_key_exists($name, $this->controller->vars)) {
|
||||||
{
|
return $this[$name];
|
||||||
return $this->graph->{$name} = $value;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function __isset($name)
|
return null;
|
||||||
{
|
}
|
||||||
return isset($this->graph->{$name});
|
|
||||||
}
|
|
||||||
|
|
||||||
|
public function __set($name, $value)
|
||||||
|
{
|
||||||
|
return $this->graph->{$name} = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function __isset($name)
|
||||||
|
{
|
||||||
|
return isset($this->graph->{$name});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+86
-88
@@ -8,111 +8,109 @@ use Cms\Classes\Controller;
|
|||||||
use Cms\Classes\CmsException;
|
use Cms\Classes\CmsException;
|
||||||
use Cms\Classes\ComponentManager;
|
use Cms\Classes\ComponentManager;
|
||||||
|
|
||||||
|
class GraphController // extends Controller
|
||||||
class GraphController// extends Controller
|
|
||||||
{
|
{
|
||||||
|
protected $graph;
|
||||||
|
|
||||||
protected $graph;
|
/**
|
||||||
|
* @var self Cache of self
|
||||||
|
*/
|
||||||
|
protected static $instance;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var self Cache of self
|
* Creates the controller.
|
||||||
*/
|
*/
|
||||||
protected static $instance;
|
public function __construct($graph, $args)
|
||||||
|
{
|
||||||
|
$this->graph = $graph;
|
||||||
|
$this->router = new GraphRouter($args);
|
||||||
|
|
||||||
|
self::$instance = $this;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates the controller.
|
* Returns an existing instance of the controller.
|
||||||
*/
|
* If the controller doesn't exists, returns null.
|
||||||
public function __construct($graph, $args)
|
* @return mixed Returns the controller object or null.
|
||||||
{
|
*/
|
||||||
$this->graph = $graph;
|
public static function getController()
|
||||||
$this->router = new GraphRouter($args);
|
{
|
||||||
|
return self::$instance;
|
||||||
|
}
|
||||||
|
|
||||||
self::$instance = $this;
|
public function component($alias)
|
||||||
|
{
|
||||||
|
if (isset($this->graph->components[$alias])) {
|
||||||
|
return $this->graph->components[$alias];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
foreach ($this->graph->settings['components'] as $component => $properties) {
|
||||||
* Returns an existing instance of the controller.
|
[$name, $component_alias] = strpos($component, ' ') ? explode(' ', $component) : [$component, $component];
|
||||||
* If the controller doesn't exists, returns null.
|
|
||||||
* @return mixed Returns the controller object or null.
|
|
||||||
*/
|
|
||||||
public static function getController()
|
|
||||||
{
|
|
||||||
return self::$instance;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function component($alias) {
|
if ($component_alias == $alias) {
|
||||||
if (isset($this->graph->components[$alias])) {
|
// make component
|
||||||
return $this->graph->components[$alias];
|
$manager = ComponentManager::instance();
|
||||||
|
|
||||||
|
if (!($componentObj = $manager->makeComponent($name, null, $properties))) {
|
||||||
|
throw new CmsException(
|
||||||
|
Lang::get('cms::lang.component.not_found', [
|
||||||
|
'name' => $name,
|
||||||
|
])
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ($this->graph->settings['components'] as $component => $properties) {
|
$this->setComponentPropertiesFromParams($componentObj, $this->router->getParameters());
|
||||||
list($name, $component_alias) = strpos($component, ' ')
|
$componentObj->init();
|
||||||
? explode(' ', $component)
|
|
||||||
: [$component, $component];
|
|
||||||
|
|
||||||
if ($component_alias == $alias) {
|
$componentObj->alias = $alias;
|
||||||
// make component
|
$this->graph->components[$alias] = $componentObj;
|
||||||
$manager = ComponentManager::instance();
|
|
||||||
|
|
||||||
if (!$componentObj = $manager->makeComponent($name, null, $properties)) {
|
return $componentObj;
|
||||||
throw new CmsException(Lang::get('cms::lang.component.not_found', ['name' => $name]));
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$this->setComponentPropertiesFromParams($componentObj, $this->router->getParameters());
|
/**
|
||||||
$componentObj->init();
|
* Sets component property values from partial parameters.
|
||||||
|
* The property values should be defined as {{ param }}.
|
||||||
|
* @param ComponentBase $component The component object.
|
||||||
|
* @param array $parameters Specifies the partial parameters.
|
||||||
|
*/
|
||||||
|
protected function setComponentPropertiesFromParams($component, $parameters = [])
|
||||||
|
{
|
||||||
|
$properties = $component->getProperties();
|
||||||
|
$routerParameters = $this->router->getParameters();
|
||||||
|
|
||||||
$componentObj->alias = $alias;
|
foreach ($properties as $propertyName => $propertyValue) {
|
||||||
$this->graph->components[$alias] = $componentObj;
|
if (is_array($propertyValue)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
return $componentObj;
|
$matches = [];
|
||||||
}
|
if (preg_match('/^\{\{([^\}]+)\}\}$/', $propertyValue, $matches)) {
|
||||||
|
$paramName = trim($matches[1]);
|
||||||
|
|
||||||
|
if (substr($paramName, 0, 1) == ':') {
|
||||||
|
$routeParamName = substr($paramName, 1);
|
||||||
|
$newPropertyValue = $routerParameters[$routeParamName] ?? null;
|
||||||
|
} else {
|
||||||
|
$newPropertyValue = $parameters[$paramName] ?? null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$component->setProperty($propertyName, $newPropertyValue);
|
||||||
|
$component->setExternalPropertyName($propertyName, $paramName);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets component property values from partial parameters.
|
* Returns a routing parameter.
|
||||||
* The property values should be defined as {{ param }}.
|
* @param string $name Routing parameter name.
|
||||||
* @param ComponentBase $component The component object.
|
* @param string $default Default to use if none is found.
|
||||||
* @param array $parameters Specifies the partial parameters.
|
* @return string
|
||||||
*/
|
*/
|
||||||
protected function setComponentPropertiesFromParams($component, $parameters = [])
|
public function param($name, $default = null)
|
||||||
{
|
{
|
||||||
$properties = $component->getProperties();
|
return $this->router->getParameter($name, $default);
|
||||||
$routerParameters = $this->router->getParameters();
|
}
|
||||||
|
|
||||||
foreach ($properties as $propertyName => $propertyValue) {
|
|
||||||
if (is_array($propertyValue)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$matches = [];
|
|
||||||
if (preg_match('/^\{\{([^\}]+)\}\}$/', $propertyValue, $matches)) {
|
|
||||||
$paramName = trim($matches[1]);
|
|
||||||
|
|
||||||
if (substr($paramName, 0, 1) == ':') {
|
|
||||||
$routeParamName = substr($paramName, 1);
|
|
||||||
$newPropertyValue = $routerParameters[$routeParamName] ?? null;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
$newPropertyValue = $parameters[$paramName] ?? null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$component->setProperty($propertyName, $newPropertyValue);
|
|
||||||
$component->setExternalPropertyName($propertyName, $paramName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns a routing parameter.
|
|
||||||
* @param string $name Routing parameter name.
|
|
||||||
* @param string $default Default to use if none is found.
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function param($name, $default = null)
|
|
||||||
{
|
|
||||||
return $this->router->getParameter($name, $default);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
+27
-30
@@ -7,38 +7,35 @@ use Cms\Classes\Router;
|
|||||||
|
|
||||||
class GraphRouter //extends Router
|
class GraphRouter //extends Router
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* @var array A list of parameters names and values extracted from the URL pattern and URL string.
|
||||||
|
*/
|
||||||
|
protected $parameters = [];
|
||||||
|
|
||||||
/**
|
public function __construct($parameters)
|
||||||
* @var array A list of parameters names and values extracted from the URL pattern and URL string.
|
{
|
||||||
*/
|
$this->parameters = $parameters;
|
||||||
protected $parameters = [];
|
}
|
||||||
|
|
||||||
public function __construct($parameters)
|
/**
|
||||||
{
|
* Returns the current routing parameters.
|
||||||
$this->parameters = $parameters;
|
* @return array
|
||||||
}
|
*/
|
||||||
|
public function getParameters()
|
||||||
/**
|
{
|
||||||
* Returns the current routing parameters.
|
return $this->parameters;
|
||||||
* @return array
|
}
|
||||||
*/
|
|
||||||
public function getParameters()
|
/**
|
||||||
{
|
* Returns a routing parameter.
|
||||||
return $this->parameters;
|
* @return array
|
||||||
}
|
*/
|
||||||
|
public function getParameter($name, $default = null)
|
||||||
|
{
|
||||||
/**
|
if (isset($this->parameters[$name]) && !empty($this->parameters[$name])) {
|
||||||
* Returns a routing parameter.
|
return $this->parameters[$name];
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public function getParameter($name, $default = null)
|
|
||||||
{
|
|
||||||
if (isset($this->parameters[$name]) && !empty($this->parameters[$name])) {
|
|
||||||
return $this->parameters[$name];
|
|
||||||
}
|
|
||||||
|
|
||||||
return $default;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return $default;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,14 +9,12 @@ use GermanAirlinesVa\Graphql\Classes\CreatesContext as GraphqlCreatesContext;
|
|||||||
|
|
||||||
use October\Rain\Support\ServiceProvider;
|
use October\Rain\Support\ServiceProvider;
|
||||||
|
|
||||||
|
class GraphqlServiceProvider extends ServiceProvider
|
||||||
class GraphqlServiceProvider extends ServiceProvider {
|
{
|
||||||
|
public function register()
|
||||||
public function register()
|
{
|
||||||
{
|
$this->app->bind(ProvidesResolver::class, GraphqlProvidesResolver::class);
|
||||||
$this->app->bind(ProvidesResolver::class, GraphqlProvidesResolver::class);
|
$this->app->singleton(CreatesContext::class, GraphqlCreatesContext::class);
|
||||||
$this->app->singleton(CreatesContext::class, GraphqlCreatesContext::class);
|
$this->app->singleton(SchemaSourceProvider::class, GraphqlSchemaSourceProvider::class);
|
||||||
$this->app->singleton(SchemaSourceProvider::class, GraphqlSchemaSourceProvider::class);
|
}
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,34 +6,32 @@ use GraphQL\Type\Definition\ResolveInfo;
|
|||||||
use Cms\Classes\CodeParser;
|
use Cms\Classes\CodeParser;
|
||||||
use Nuwave\Lighthouse\Schema\ResolverProvider as LighthouseResolverProvider;
|
use Nuwave\Lighthouse\Schema\ResolverProvider as LighthouseResolverProvider;
|
||||||
|
|
||||||
|
class ResolverProvider extends LighthouseResolverProvider
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Provide a field resolver in case no resolver directive is defined for a field.
|
||||||
|
*
|
||||||
|
* @param \Nuwave\Lighthouse\Schema\Values\FieldValue $fieldValue
|
||||||
|
* @return \Closure
|
||||||
|
*/
|
||||||
|
public function provideResolver(FieldValue $fieldValue): Closure
|
||||||
|
{
|
||||||
|
return function ($root, array $args, $context, ResolveInfo $resolveInfo) use ($fieldValue) {
|
||||||
|
$fieldName = $fieldValue->getFieldName();
|
||||||
|
// use local graph resolver in code section if existent
|
||||||
|
if ($graphObj = $context->source->findGraph($fieldName)) {
|
||||||
|
/* @var $graphObj \GermanAirlinesVa\Graphql\Classes\Graph */
|
||||||
|
$parser = new CodeParser($graphObj);
|
||||||
|
$codeObj = $parser->source($graphObj, null, new GraphController($graphObj, $args));
|
||||||
|
$resolveMethod = 'resolve' . studly_case($fieldName);
|
||||||
|
|
||||||
class ResolverProvider extends LighthouseResolverProvider {
|
if (method_exists($codeObj, $resolveMethod)) {
|
||||||
|
return $codeObj->$resolveMethod($root, $args, $context, $resolveInfo);
|
||||||
/**
|
}
|
||||||
* Provide a field resolver in case no resolver directive is defined for a field.
|
}
|
||||||
*
|
|
||||||
* @param \Nuwave\Lighthouse\Schema\Values\FieldValue $fieldValue
|
|
||||||
* @return \Closure
|
|
||||||
*/
|
|
||||||
public function provideResolver(FieldValue $fieldValue): Closure
|
|
||||||
{
|
|
||||||
return function ($root, array $args, $context, ResolveInfo $resolveInfo) use ($fieldValue) {
|
|
||||||
$fieldName = $fieldValue->getFieldName();
|
|
||||||
// use local graph resolver in code section if existent
|
|
||||||
if ($graphObj = $context->source->findGraph($fieldName)) {
|
|
||||||
/* @var $graphObj \GermanAirlinesVa\Graphql\Classes\Graph */
|
|
||||||
$parser = new CodeParser($graphObj);
|
|
||||||
$codeObj = $parser->source($graphObj, null, new GraphController($graphObj, $args));
|
|
||||||
$resolveMethod = 'resolve' . studly_case($fieldName);
|
|
||||||
|
|
||||||
if (method_exists($codeObj, $resolveMethod)) {
|
|
||||||
return $codeObj->$resolveMethod($root, $args, $context, $resolveInfo);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// default resolver
|
|
||||||
return parent::provideResolver($fieldValue)($root, $args, $context, $resolveInfo);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// default resolver
|
||||||
|
return parent::provideResolver($fieldValue)($root, $args, $context, $resolveInfo);
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+30
-32
@@ -6,39 +6,37 @@ use Cms\Classes\Theme;
|
|||||||
|
|
||||||
class Schema extends Theme
|
class Schema extends Theme
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
/**
|
* Returns the absolute theme path.
|
||||||
* Returns the absolute theme path.
|
* @param string $dirName Optional theme directory. Defaults to $this->getDirName()
|
||||||
* @param string $dirName Optional theme directory. Defaults to $this->getDirName()
|
* @return string
|
||||||
* @return string
|
*/
|
||||||
*/
|
public function getPath(?string $dirName = null): string
|
||||||
public function getPath(?string $dirName = null): string
|
{
|
||||||
{
|
if (!$dirName) {
|
||||||
if (!$dirName) {
|
$dirName = $this->getDirName();
|
||||||
$dirName = $this->getDirName();
|
|
||||||
}
|
|
||||||
return base_path() . '/' . $dirName;
|
|
||||||
}
|
}
|
||||||
|
return base_path() . '/' . $dirName;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a list of graphs in the template.
|
* Returns a list of graphs in the template.
|
||||||
* @param boolean $skipCache Indicates if the pages should be reloaded from the disk bypassing the cache.
|
* @param boolean $skipCache Indicates if the pages should be reloaded from the disk bypassing the cache.
|
||||||
* @return array Returns an array of Nocio\Headstart\Classes\Graph objects.
|
* @return array Returns an array of Nocio\Headstart\Classes\Graph objects.
|
||||||
*/
|
*/
|
||||||
public function listGraphs($skipCache = false)
|
public function listGraphs($skipCache = false)
|
||||||
{
|
{
|
||||||
return Graph::listInTheme($this, $skipCache);
|
return Graph::listInTheme($this, $skipCache);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the active theme code.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
* If the theme doesn't exist, returns null.
|
|
||||||
*/
|
|
||||||
public static function getActiveThemeCode(): ?string
|
|
||||||
{
|
|
||||||
return 'graphql';
|
|
||||||
}
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the active theme code.
|
||||||
|
*
|
||||||
|
* @return string
|
||||||
|
* If the theme doesn't exist, returns null.
|
||||||
|
*/
|
||||||
|
public static function getActiveThemeCode(): ?string
|
||||||
|
{
|
||||||
|
return 'graphql';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+12
-16
@@ -7,24 +7,20 @@ use Nuwave\Lighthouse\Schema\Context;
|
|||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
use Nuwave\Lighthouse\Schema\Source\SchemaSourceProvider;
|
use Nuwave\Lighthouse\Schema\Source\SchemaSourceProvider;
|
||||||
|
|
||||||
|
|
||||||
class SchemaContext extends Context
|
class SchemaContext extends Context
|
||||||
{
|
{
|
||||||
|
public $source;
|
||||||
|
|
||||||
public $source;
|
/**
|
||||||
|
* Create new context.
|
||||||
/**
|
*
|
||||||
* Create new context.
|
* @param Request $request
|
||||||
*
|
* @param $source
|
||||||
* @param Request $request
|
*/
|
||||||
* @param $source
|
public function __construct(Request $request)
|
||||||
*/
|
{
|
||||||
public function __construct(Request $request)
|
parent::__construct($request);
|
||||||
{
|
|
||||||
parent::__construct($request);
|
|
||||||
|
|
||||||
$this->source = App::make(SchemaSourceProvider::class);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
$this->source = App::make(SchemaSourceProvider::class);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+171
-171
@@ -11,211 +11,211 @@ use Cms\Classes\ComponentPartial;
|
|||||||
|
|
||||||
class SchemaSourceProvider implements LighthouseSchemaSourceProvider
|
class SchemaSourceProvider implements LighthouseSchemaSourceProvider
|
||||||
{
|
{
|
||||||
|
protected $fieldGraphMapCacheKey = 'graphql-schema-field-graph-map';
|
||||||
|
|
||||||
protected $fieldGraphMapCacheKey = 'graphql-schema-field-graph-map';
|
/**
|
||||||
|
* @var Schema
|
||||||
|
*/
|
||||||
|
public $template;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var Schema
|
* @var string
|
||||||
*/
|
*/
|
||||||
public $template;
|
protected $rootSchemaPath;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var string
|
* @var array
|
||||||
*/
|
* Collects graphs and their parsed schema
|
||||||
protected $rootSchemaPath;
|
*/
|
||||||
|
protected $graphMap;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var array
|
* @var array
|
||||||
* Collects graphs and their parsed schema
|
* Maps field names to graphs
|
||||||
*/
|
*/
|
||||||
protected $graphMap;
|
protected $fieldGraphMap;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @var array
|
* SchemaSource constructor.
|
||||||
* Maps field names to graphs
|
*
|
||||||
*/
|
* @param string|null $template
|
||||||
protected $fieldGraphMap;
|
*/
|
||||||
|
public function __construct($template = null)
|
||||||
|
{
|
||||||
|
$this->template = Schema::load(is_string($template) ? $template : 'graphql');
|
||||||
|
$this->rootSchemaPath = $this->template->getPath();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set schema root path.
|
||||||
|
*
|
||||||
|
* @param string $path
|
||||||
|
*
|
||||||
|
* @return SchemaSourceProvider
|
||||||
|
*/
|
||||||
|
public function setRootPath(string $path): self
|
||||||
|
{
|
||||||
|
$this->rootSchemaPath = $path;
|
||||||
|
|
||||||
/**
|
return $this;
|
||||||
* SchemaSource constructor.
|
}
|
||||||
*
|
|
||||||
* @param string|null $template
|
|
||||||
*/
|
|
||||||
public function __construct($template = null)
|
|
||||||
{
|
|
||||||
$this->template = Schema::load(is_string($template) ? $template : 'graphql');
|
|
||||||
$this->rootSchemaPath = $this->template->getPath();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set schema root path.
|
* Stitch together schema documents and return the result as a string.
|
||||||
*
|
*
|
||||||
* @param string $path
|
* @return string
|
||||||
*
|
*/
|
||||||
* @return SchemaSourceProvider
|
public function getSchemaString(): string
|
||||||
*/
|
{
|
||||||
public function setRootPath(string $path): self
|
// root types
|
||||||
{
|
$schema = "
|
||||||
$this->rootSchemaPath = $path;
|
|
||||||
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Stitch together schema documents and return the result as a string.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function getSchemaString(): string
|
|
||||||
{
|
|
||||||
// root types
|
|
||||||
$schema = '
|
|
||||||
type Query {
|
type Query {
|
||||||
graphql: Boolean
|
dummy: Boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
type Mutation {
|
type Mutation {
|
||||||
graphql: Boolean
|
dummy: Boolean
|
||||||
}
|
}
|
||||||
';
|
|
||||||
// schema
|
type Subscription {
|
||||||
$schema .= collect($this->getGraphMap())->implode('schema', '');
|
dummy: Boolean @subscription(class: \"GermanAirlinesVa\\\\Graphql\\\\Classes\\\\Dummy\")
|
||||||
return $schema;
|
}
|
||||||
|
";
|
||||||
|
// schema
|
||||||
|
$schema .= collect($this->getGraphMap())->implode('schema', '');
|
||||||
|
return $schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function findGraph($fieldName, $findOrFail = false)
|
||||||
|
{
|
||||||
|
if (!isset($this->getFieldGraphMap()[$fieldName])) {
|
||||||
|
if ($findOrFail) {
|
||||||
|
throw new Error("Could not find graph of field '{$fieldName}'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function findGraph($fieldName, $findOrFail=false) {
|
$name = $this->getFieldGraphMap()[$fieldName];
|
||||||
if (!isset($this->getFieldGraphMap()[$fieldName])) {
|
|
||||||
if ($findOrFail) {
|
|
||||||
throw new Error("Could not find graph of field '{$fieldName}'.");
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
if (isset($this->graphMap[$name])) {
|
||||||
}
|
// if available, use already instantiated graph objects in memory
|
||||||
|
return $this->graphMap[$name]['graph'];
|
||||||
$name = $this->getFieldGraphMap()[$fieldName];
|
|
||||||
|
|
||||||
if (isset($this->graphMap[$name])) {
|
|
||||||
// if available, use already instantiated graph objects in memory
|
|
||||||
return $this->graphMap[$name]['graph'];
|
|
||||||
}
|
|
||||||
|
|
||||||
return Graph::loadCached($this->template, $name);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getFieldGraphMap() {
|
return Graph::loadCached($this->template, $name);
|
||||||
if (! is_null($this->fieldGraphMap)) {
|
}
|
||||||
return $this->fieldGraphMap;
|
|
||||||
}
|
|
||||||
|
|
||||||
// restore from cache
|
public function getFieldGraphMap()
|
||||||
|
{
|
||||||
|
if (!is_null($this->fieldGraphMap)) {
|
||||||
|
return $this->fieldGraphMap;
|
||||||
|
}
|
||||||
|
|
||||||
$cacheable = true;
|
// restore from cache
|
||||||
|
|
||||||
if ($cacheable) {
|
$cacheable = true;
|
||||||
$map = Cache::get($this->fieldGraphMapCacheKey, false);
|
|
||||||
if (
|
|
||||||
$map &&
|
|
||||||
($map = @unserialize(@base64_decode($map))) &&
|
|
||||||
is_array($map)
|
|
||||||
) {
|
|
||||||
$this->fieldGraphMap = $map;
|
|
||||||
return $this->fieldGraphMap;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// rebuild mapping
|
|
||||||
|
|
||||||
$this->fieldGraphMap = [];
|
|
||||||
|
|
||||||
foreach ($this->getGraphMap() as $key => $element) {
|
|
||||||
if (empty(trim($element['schema']))) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// parse the AST to collect fields mapping
|
|
||||||
$ast = Parser::parse($element['schema'], ['noLocation' => true]);
|
|
||||||
$fields = collect($ast->definitions)
|
|
||||||
->filter(function ($node) {
|
|
||||||
// we only consider the root types since their fields are unique
|
|
||||||
return $node->name->value == 'Query' || $node->name->value == 'Mutation';
|
|
||||||
})
|
|
||||||
->flatMap(function ($node) {
|
|
||||||
// collect field names
|
|
||||||
return collect($node->fields)->map(function($field) {
|
|
||||||
return $field->name->value;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// map as [$field_name => $graph_filename]
|
|
||||||
$this->fieldGraphMap = array_merge($this->fieldGraphMap, array_fill_keys($fields->toArray(), $key));
|
|
||||||
}
|
|
||||||
|
|
||||||
// cache mapping
|
|
||||||
|
|
||||||
if ($cacheable) {
|
|
||||||
Cache::put(
|
|
||||||
$this->fieldGraphMapCacheKey,
|
|
||||||
base64_encode(serialize($this->fieldGraphMap)),
|
|
||||||
10
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
if ($cacheable) {
|
||||||
|
$map = Cache::get($this->fieldGraphMapCacheKey, false);
|
||||||
|
if ($map && ($map = @unserialize(@base64_decode($map))) && is_array($map)) {
|
||||||
|
$this->fieldGraphMap = $map;
|
||||||
return $this->fieldGraphMap;
|
return $this->fieldGraphMap;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getGraphMap() {
|
// rebuild mapping
|
||||||
if (!is_null($this->graphMap)) {
|
|
||||||
return $this->graphMap;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->graphMap = [];
|
$this->fieldGraphMap = [];
|
||||||
$component_re = '/{%\s*component\s+["|\'](.*)["|\']\s*%}/m';
|
|
||||||
$component_str = [$this, 'getComponentSchemaString'];
|
|
||||||
foreach ($this->template->listGraphs() as $graph) {
|
|
||||||
/* @var $graph \GermanAirlinesVa\Graphql\Classes\Graph */
|
|
||||||
$markup = $graph->markup;
|
|
||||||
|
|
||||||
$schema = preg_replace_callback($component_re, function(array $matches) use ($graph, $component_str) {
|
foreach ($this->getGraphMap() as $key => $element) {
|
||||||
$matched_alias = $matches[1];
|
if (empty(trim($element['schema']))) {
|
||||||
foreach ($graph->settings['components'] as $component => $properties) {
|
continue;
|
||||||
// find component by alias
|
}
|
||||||
list($name, $alias) = strpos($component, ' ')
|
|
||||||
? explode(' ', $component)
|
|
||||||
: [$component, $component];
|
|
||||||
if ($alias == $matched_alias) {
|
|
||||||
// resolve component schema
|
|
||||||
return $component_str($name, $alias);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// if not found, remove
|
|
||||||
return '';
|
|
||||||
}, $markup);
|
|
||||||
|
|
||||||
$this->graphMap[$graph->getFileName()] = [
|
// parse the AST to collect fields mapping
|
||||||
'graph' => $graph,
|
$ast = Parser::parse($element['schema'], ['noLocation' => true]);
|
||||||
'schema' => $schema
|
$fields = collect($ast->definitions)
|
||||||
];
|
->filter(function ($node) {
|
||||||
}
|
// we only consider the root types since their fields are unique
|
||||||
|
return $node->name->value == 'Query' || $node->name->value == 'Mutation';
|
||||||
|
})
|
||||||
|
->flatMap(function ($node) {
|
||||||
|
// collect field names
|
||||||
|
return collect($node->fields)->map(function ($field) {
|
||||||
|
return $field->name->value;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
return $this->graphMap;
|
// map as [$field_name => $graph_filename]
|
||||||
|
$this->fieldGraphMap = array_merge($this->fieldGraphMap, array_fill_keys($fields->toArray(), $key));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getComponentSchemaString($componentName, $alias) {
|
// cache mapping
|
||||||
$manager = ComponentManager::instance();
|
|
||||||
$componentObj = $manager->makeComponent($componentName);
|
|
||||||
if ($partial = ComponentPartial::load($componentObj, 'schema.graphqls')) {
|
|
||||||
$content = $partial->getContent();
|
|
||||||
$content = str_replace('__SELF__', $alias, $content);
|
|
||||||
|
|
||||||
return $content;
|
if ($cacheable) {
|
||||||
} else {
|
Cache::put($this->fieldGraphMapCacheKey, base64_encode(serialize($this->fieldGraphMap)), 10);
|
||||||
return "# {$componentName} does not provide a default schema definition";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function clearCache() {
|
return $this->fieldGraphMap;
|
||||||
Cache::forget($this->fieldGraphMapCacheKey);
|
}
|
||||||
|
|
||||||
|
public function getGraphMap()
|
||||||
|
{
|
||||||
|
if (!is_null($this->graphMap)) {
|
||||||
|
return $this->graphMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$this->graphMap = [];
|
||||||
|
$component_re = '/{%\s*component\s+["|\'](.*)["|\']\s*%}/m';
|
||||||
|
$component_str = [$this, 'getComponentSchemaString'];
|
||||||
|
foreach ($this->template->listGraphs() as $graph) {
|
||||||
|
/* @var $graph \GermanAirlinesVa\Graphql\Classes\Graph */
|
||||||
|
$markup = $graph->markup;
|
||||||
|
|
||||||
|
$schema = preg_replace_callback(
|
||||||
|
$component_re,
|
||||||
|
function (array $matches) use ($graph, $component_str) {
|
||||||
|
$matched_alias = $matches[1];
|
||||||
|
foreach ($graph->settings['components'] as $component => $properties) {
|
||||||
|
// find component by alias
|
||||||
|
[$name, $alias] = strpos($component, ' ') ? explode(' ', $component) : [$component, $component];
|
||||||
|
if ($alias == $matched_alias) {
|
||||||
|
// resolve component schema
|
||||||
|
return $component_str($name, $alias);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// if not found, remove
|
||||||
|
return '';
|
||||||
|
},
|
||||||
|
$markup
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->graphMap[$graph->getFileName()] = [
|
||||||
|
'graph' => $graph,
|
||||||
|
'schema' => $schema,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $this->graphMap;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getComponentSchemaString($componentName, $alias)
|
||||||
|
{
|
||||||
|
$manager = ComponentManager::instance();
|
||||||
|
$componentObj = $manager->makeComponent($componentName);
|
||||||
|
if ($partial = ComponentPartial::load($componentObj, 'schema.graphqls')) {
|
||||||
|
$content = $partial->getContent();
|
||||||
|
$content = str_replace('__SELF__', $alias, $content);
|
||||||
|
|
||||||
|
return $content;
|
||||||
|
} else {
|
||||||
|
return "# {$componentName} does not provide a default schema definition";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function clearCache()
|
||||||
|
{
|
||||||
|
Cache::forget($this->fieldGraphMapCacheKey);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+120
-104
@@ -1,19 +1,37 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
use Cms\Classes\Theme;
|
use Cms\Classes\Theme;
|
||||||
|
use GermanAirlinesVa\Graphql\Models\Settings;
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'packages' => [
|
'connection' => [
|
||||||
'nuwave/lighthouse' => [
|
'driver' => 'mysql',
|
||||||
'config_namespace' => 'lighthouse',
|
'url' => env('DATABASE_URL'),
|
||||||
'providers' => [
|
'host' => env('DB_HOST', '127.0.0.1'),
|
||||||
'Nuwave\Lighthouse\LighthouseServiceProvider'
|
'port' => env('DB_PORT', '3306'),
|
||||||
],
|
'database' => 'germanairlinesva_graphql',
|
||||||
'aliases' => [
|
'username' => env('DB_USERNAME', 'root'),
|
||||||
],
|
'password' => env('DB_PASSWORD', ''),
|
||||||
'config' => [
|
'unix_socket' => env('DB_SOCKET', ''),
|
||||||
|
'charset' => 'utf8mb4',
|
||||||
/*
|
'collation' => 'utf8mb4_unicode_ci',
|
||||||
|
'prefix' => '',
|
||||||
|
'prefix_indexes' => true,
|
||||||
|
'strict' => true,
|
||||||
|
'engine' => 'InnoDB',
|
||||||
|
'options' => extension_loaded('pdo_mysql')
|
||||||
|
? array_filter([
|
||||||
|
PDO::MYSQL_ATTR_SSL_CA => env('MYSQL_ATTR_SSL_CA'),
|
||||||
|
])
|
||||||
|
: [],
|
||||||
|
],
|
||||||
|
'packages' => [
|
||||||
|
'nuwave/lighthouse' => [
|
||||||
|
'config_namespace' => 'lighthouse',
|
||||||
|
'providers' => ['Nuwave\Lighthouse\LighthouseServiceProvider'],
|
||||||
|
'aliases' => [],
|
||||||
|
'config' => [
|
||||||
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Route Configuration
|
| Route Configuration
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -24,27 +42,27 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'route' => [
|
'route' => [
|
||||||
/*
|
/*
|
||||||
* The URI the endpoint responds to, e.g. mydomain.com/graphql.
|
* The URI the endpoint responds to, e.g. mydomain.com/graphql.
|
||||||
*/
|
*/
|
||||||
'uri' => 'graphql',
|
'uri' => 'graphql',
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Lighthouse creates a named route for convenient URL generation and redirects.
|
* Lighthouse creates a named route for convenient URL generation and redirects.
|
||||||
*/
|
*/
|
||||||
'name' => 'graphql',
|
'name' => 'graphql',
|
||||||
|
|
||||||
/*
|
/*
|
||||||
*
|
*
|
||||||
* Beware that middleware defined here runs before the GraphQL execution phase,
|
* Beware that middleware defined here runs before the GraphQL execution phase,
|
||||||
* so you have to take extra care to return spec-compliant error responses.
|
* so you have to take extra care to return spec-compliant error responses.
|
||||||
* To apply middleware on a field level, use the @middleware directive.
|
* To apply middleware on a field level, use the @middleware directive.
|
||||||
*/
|
*/
|
||||||
'middleware' => [\Nuwave\Lighthouse\Support\Http\Middleware\AcceptJson::class]
|
'middleware' => [\Nuwave\Lighthouse\Support\Http\Middleware\AcceptJson::class],
|
||||||
],
|
],
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Schema Declaration
|
| Schema Declaration
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -55,11 +73,11 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'schema' => [
|
'schema' => [
|
||||||
'register' => '',
|
'register' => '',
|
||||||
],
|
],
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Schema Cache
|
| Schema Cache
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -70,13 +88,13 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'cache' => [
|
'cache' => [
|
||||||
'enable' => false,
|
'enable' => Settings::get('enable_cache', false),
|
||||||
'key' => env('GRAPHQL_CACHE_KEY', 'graphql-schema'),
|
'key' => env('GRAPHQL_CACHE_KEY', 'graphql-schema'),
|
||||||
'ttl' => env('GRAPHQL_CACHE_TTL', null),
|
'ttl' => env('GRAPHQL_CACHE_TTL', null),
|
||||||
],
|
],
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Namespaces
|
| Namespaces
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -87,18 +105,18 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'namespaces' => [
|
'namespaces' => [
|
||||||
'models' => 'Graphql\\Models',
|
'models' => 'Graphql\\Models',
|
||||||
'queries' => 'Graphql\\Queries',
|
'queries' => 'Graphql\\Queries',
|
||||||
'mutations' => 'Graphql\\Mutations',
|
'mutations' => 'Graphql\\Mutations',
|
||||||
'subscriptions' => 'Graphql\\Subscriptions',
|
'subscriptions' => 'Graphql\\Subscriptions',
|
||||||
'interfaces' => 'Graphql\\Interfaces',
|
'interfaces' => 'Graphql\\Interfaces',
|
||||||
'unions' => 'Graphql\\Unions',
|
'unions' => 'Graphql\\Unions',
|
||||||
'scalars' => 'Graphql\\Scalars',
|
'scalars' => 'Graphql\\Scalars',
|
||||||
'directives' => ['GermanAirlinesVa\\Graphql\\GraphQL\\Directives', 'Graphql\\Directives'],
|
'directives' => ['GermanAirlinesVa\\Graphql\\GraphQL\\Directives', 'Graphql\\Directives'],
|
||||||
],
|
],
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Security
|
| Security
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -108,13 +126,13 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'security' => [
|
'security' => [
|
||||||
'max_query_complexity' => \GraphQL\Validator\Rules\QueryComplexity::DISABLED,
|
'max_query_complexity' => \GraphQL\Validator\Rules\QueryComplexity::DISABLED,
|
||||||
'max_query_depth' => \GraphQL\Validator\Rules\QueryDepth::DISABLED,
|
'max_query_depth' => \GraphQL\Validator\Rules\QueryDepth::DISABLED,
|
||||||
'disable_introspection' => \GraphQL\Validator\Rules\DisableIntrospection::DISABLED,
|
'disable_introspection' => \GraphQL\Validator\Rules\DisableIntrospection::DISABLED,
|
||||||
],
|
],
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Pagination
|
| Pagination
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -125,9 +143,9 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'paginate_max_count' => null,
|
'paginate_max_count' => null,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Pagination Amount Argument
|
| Pagination Amount Argument
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -138,9 +156,9 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'pagination_amount_argument' => 'first',
|
'pagination_amount_argument' => 'first',
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Debug
|
| Debug
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -150,9 +168,9 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'debug' => \GraphQL\Error\Debug::INCLUDE_DEBUG_MESSAGE | \GraphQL\Error\Debug::INCLUDE_TRACE,
|
//'debug' => \GraphQL\Error\Debug::INCLUDE_DEBUG_MESSAGE | \GraphQL\Error\Debug::INCLUDE_TRACE,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Error Handlers
|
| Error Handlers
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -163,11 +181,9 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'error_handlers' => [
|
'error_handlers' => [\Nuwave\Lighthouse\Execution\ExtensionErrorHandler::class],
|
||||||
\Nuwave\Lighthouse\Execution\ExtensionErrorHandler::class,
|
|
||||||
],
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Global ID
|
| Global ID
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -177,9 +193,9 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'global_id_field' => 'id',
|
'global_id_field' => 'id',
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Batched Queries
|
| Batched Queries
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -189,9 +205,9 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'batched_queries' => true,
|
'batched_queries' => Settings::get('batched_queries', true),
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| Transactional Mutations
|
| Transactional Mutations
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -201,9 +217,9 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'transactional_mutations' => true,
|
'transactional_mutations' => true,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| GraphQL Subscriptions
|
| GraphQL Subscriptions
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
@@ -213,39 +229,39 @@ return [
|
|||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
'subscriptions' => [
|
'subscriptions' => [
|
||||||
/*
|
/*
|
||||||
* Determines if broadcasts should be queued by default.
|
* Determines if broadcasts should be queued by default.
|
||||||
*/
|
*/
|
||||||
'queue_broadcasts' => env('LIGHTHOUSE_QUEUE_BROADCASTS', true),
|
'queue_broadcasts' => env('LIGHTHOUSE_QUEUE_BROADCASTS', true),
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Default subscription storage.
|
* Default subscription storage.
|
||||||
*
|
*
|
||||||
* Any Laravel supported cache driver options are available here.
|
* Any Laravel supported cache driver options are available here.
|
||||||
*/
|
*/
|
||||||
'storage' => env('LIGHTHOUSE_SUBSCRIPTION_STORAGE', 'redis'),
|
'storage' => env('LIGHTHOUSE_SUBSCRIPTION_STORAGE', 'redis'),
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Default subscription broadcaster.
|
* Default subscription broadcaster.
|
||||||
*/
|
*/
|
||||||
'broadcaster' => env('LIGHTHOUSE_BROADCASTER', 'pusher'),
|
'broadcaster' => env('LIGHTHOUSE_BROADCASTER', 'pusher'),
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Subscription broadcasting drivers with config options.
|
* Subscription broadcasting drivers with config options.
|
||||||
*/
|
*/
|
||||||
'broadcasters' => [
|
'broadcasters' => [
|
||||||
'log' => [
|
'log' => [
|
||||||
'driver' => 'log',
|
'driver' => 'log',
|
||||||
],
|
],
|
||||||
'pusher' => [
|
'pusher' => [
|
||||||
'driver' => 'pusher',
|
'driver' => 'pusher',
|
||||||
'routes' => \Nuwave\Lighthouse\Subscriptions\SubscriptionRouter::class.'@pusher',
|
'routes' => \Nuwave\Lighthouse\Subscriptions\SubscriptionRouter::class . '@pusher',
|
||||||
'connection' => 'pusher',
|
'connection' => 'pusher',
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
],
|
],
|
||||||
]
|
],
|
||||||
]
|
],
|
||||||
]
|
],
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -0,0 +1,17 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\Graphql\Controllers;
|
||||||
|
|
||||||
|
use Backend\Classes\Controller;
|
||||||
|
|
||||||
|
class GraphqlKey extends Controller
|
||||||
|
{
|
||||||
|
public $implement = ['Backend\Behaviors\ListController'];
|
||||||
|
|
||||||
|
public $listConfig = 'config_list.yaml';
|
||||||
|
|
||||||
|
public $requiredPermissions = ['germanairlinesva.graphql.master'];
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\Graphql\Controllers;
|
||||||
|
|
||||||
|
use Backend\Classes\Controller;
|
||||||
|
|
||||||
|
class Playground extends Controller
|
||||||
|
{
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct();
|
||||||
|
}
|
||||||
|
|
||||||
|
public $requiredPermissions = ['germanairlinesva.graphql.master'];
|
||||||
|
|
||||||
|
public function index()
|
||||||
|
{
|
||||||
|
$this->pageTitle = 'germanairlinesva.graphql::lang.menu.playground';
|
||||||
|
$this->pageTitleTemplate = '%s ' . trans($this->pageTitle);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
<div data-control="toolbar">
|
||||||
|
<button
|
||||||
|
class="btn btn-default oc-icon-trash-o"
|
||||||
|
disabled="disabled"
|
||||||
|
onclick="$(this).data('request-data', {
|
||||||
|
checked: $('.control-list').listWidget('getChecked')
|
||||||
|
})"
|
||||||
|
data-request="onDelete"
|
||||||
|
data-request-confirm="<?= e(trans('backend::lang.list.delete_selected_confirm')) ?>"
|
||||||
|
data-trigger-action="enable"
|
||||||
|
data-trigger=".control-list input[type=checkbox]"
|
||||||
|
data-trigger-condition="checked"
|
||||||
|
data-request-success="$(this).prop('disabled', true)"
|
||||||
|
data-stripe-load-indicator
|
||||||
|
>
|
||||||
|
<?= e(trans('backend::lang.list.delete_selected')) ?>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
list: $/germanairlinesva/graphql/models/graphqlkey/columns.yaml
|
||||||
|
modelClass: GermanAirlinesVa\Graphql\Models\GraphqlKey
|
||||||
|
title: GraphQL Keys
|
||||||
|
noRecordsMessage: 'backend::lang.list.no_records'
|
||||||
|
showSetup: true
|
||||||
|
showCheckboxes: true
|
||||||
|
recordsPerPage: 20
|
||||||
|
toolbar:
|
||||||
|
buttons: list_toolbar
|
||||||
|
search:
|
||||||
|
prompt: 'backend::lang.list.search_prompt'
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
<?= $this->listRender() ?>
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
<div id="mount"></div>
|
||||||
|
<script type="text/javascript" src="/plugins/germanairlinesva/graphql/assets/js/graphiql.js"></script>
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
==
|
||||||
|
<?php
|
||||||
|
use Cms\Classes\Theme;
|
||||||
|
use Cms\Classes\Content;
|
||||||
|
|
||||||
|
function resolveCmsContent($root, $args) {
|
||||||
|
if (! $theme = Theme::getActiveTheme()) {
|
||||||
|
throw new \Exception(Lang::get('cms::lang.theme.active.not_found'));
|
||||||
|
}
|
||||||
|
|
||||||
|
return Content::loadCached($theme, $args['name']);
|
||||||
|
}
|
||||||
|
?>
|
||||||
|
==
|
||||||
|
# Content
|
||||||
|
extend type Query {
|
||||||
|
cmsContent(name: String!): CmsContent!
|
||||||
|
}
|
||||||
|
|
||||||
|
type CmsContent {
|
||||||
|
fileName: String!
|
||||||
|
content: String
|
||||||
|
markup: String
|
||||||
|
parsedMarkup: String
|
||||||
|
}
|
||||||
|
|
||||||
|
# CMS-wide types
|
||||||
|
# Backend\Models\User
|
||||||
|
type BackendUser { id: ID! }
|
||||||
|
#System\Models\File
|
||||||
|
type SystemFile { id: ID path: String title: String description: String }
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
==
|
||||||
|
==
|
||||||
|
# Content
|
||||||
|
extend type Query {
|
||||||
|
aircraft: [Aircraft] @all(model: "GermanAirlinesVa\\Fleet\\Models\\Aircraft")
|
||||||
|
aircraftType: [AircraftType] @all(model: "GermanAirlinesVa\\Fleet\\Models\\AircraftType")
|
||||||
|
}
|
||||||
|
extend type Subscription {
|
||||||
|
aircraftAdded: Aircraft @subscription(class: "GermanAirlinesVa\\Fleet\\Classes\\AircraftAdded")
|
||||||
|
}
|
||||||
|
extend type Mutation {
|
||||||
|
addAircraft(aircraft_type_id: ID!, home_airport_id: ID!, name: String!, registration: String! ): Aircraft
|
||||||
|
@create(model:"GermanAirlinesVa\\Fleet\\Models\\Aircraft")
|
||||||
|
@broadcast(subscription: "aircraftAdded")
|
||||||
|
}
|
||||||
|
|
||||||
|
type Aircraft {
|
||||||
|
id: ID!
|
||||||
|
aircraft_type_id: ID!
|
||||||
|
home_airport_id: ID!
|
||||||
|
name: String!
|
||||||
|
registration: String!
|
||||||
|
}
|
||||||
|
type AircraftType {
|
||||||
|
type: String!
|
||||||
|
aircrafts: [Aircraft] @hasMany
|
||||||
|
}
|
||||||
@@ -0,0 +1,150 @@
|
|||||||
|
==
|
||||||
|
<?php
|
||||||
|
function resolveExamMemberRanks($root, $args) {
|
||||||
|
$exams = \GermanAirlinesVa\Schooling\Models\ExamMemberRank::where('member_id', $args['memberID'])->get();
|
||||||
|
foreach($exams as $exam) {
|
||||||
|
$exam->load([
|
||||||
|
'exam_questions' => function ($query) {
|
||||||
|
$query->groupBy('id');
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
return $exams;
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveExamTyperatings($root, $args) {
|
||||||
|
$exams = \GermanAirlinesVa\Schooling\Models\ExamTyperating::where('member_id', $args['memberID'])->get();
|
||||||
|
foreach($exams as $exam) {
|
||||||
|
$exam->load([
|
||||||
|
'exam_questions' => function ($query) {
|
||||||
|
$query->groupBy('id');
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
return $exams;
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveTyperatings() {
|
||||||
|
$typeratings = GermanAirlinesVa\Schooling\Models\Typerating::all();
|
||||||
|
foreach($typeratings as $typerating) {
|
||||||
|
$typerating->load([
|
||||||
|
'exam_questions' => function($query) {
|
||||||
|
$query->groupBy('id');
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
return $typeratings;
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveMemberRanks() {
|
||||||
|
$ranks = GermanAirlinesVa\Schooling\Models\MemberRank::all();
|
||||||
|
foreach($ranks as $rank) {
|
||||||
|
$rank->load([
|
||||||
|
'exam_questions' => function($query) {
|
||||||
|
$query->groupBy('id');
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
return $rank;
|
||||||
|
}
|
||||||
|
|
||||||
|
function createExamMemberRank($root, $args) {
|
||||||
|
return GermanAirlinesVa\Schooling\Models\ExamMemberRank::createNew($args['memberID'], $args['memberRankID']);
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveCreateExamTyperating($root, $args) {
|
||||||
|
return GermanAirlinesVa\Schooling\Models\ExamTyperating::createNew($args['memberID'], $args['typeratingID']);
|
||||||
|
}
|
||||||
|
==
|
||||||
|
# Content
|
||||||
|
extend type Query {
|
||||||
|
memberRanks: [MemberRank]!
|
||||||
|
typeratings: [Typerating]!
|
||||||
|
examMemberRanks(memberID: ID!): [ExamMemberRank]!
|
||||||
|
examTyperatings(memberID: ID!): [ExamTyperating]!
|
||||||
|
}
|
||||||
|
extend type Mutation {
|
||||||
|
createExamMemberRank(memberID: ID!, memberRankID: ID!): ExamMemberRank!
|
||||||
|
createExamTyperating(memberID: ID!, typeratingID: ID!): ExamTyperating!
|
||||||
|
}
|
||||||
|
|
||||||
|
type MemberRank {
|
||||||
|
id: ID!
|
||||||
|
name: String!
|
||||||
|
description: String!
|
||||||
|
points: Int!
|
||||||
|
price: Int!
|
||||||
|
badge: String!
|
||||||
|
region: String!
|
||||||
|
exam_questions: [ExamMemberRankQuestion]! @hasMany
|
||||||
|
exams: [ExamMemberRank]! @hasMany
|
||||||
|
}
|
||||||
|
type Typerating {
|
||||||
|
id: ID!
|
||||||
|
name: String!
|
||||||
|
price: Int!
|
||||||
|
exam_questions: [ExamTyperatingQuestion]! @hasMany
|
||||||
|
exams: [ExamTyperating]! @hasMany
|
||||||
|
}
|
||||||
|
type ExamMemberRank {
|
||||||
|
id: ID!
|
||||||
|
member_id: ID!
|
||||||
|
member_rank: MemberRank! @belongsTo
|
||||||
|
start: DateTime!
|
||||||
|
status: ExamStatus!
|
||||||
|
exam_questions: [ExamMemberRankQuestion]! @belongsToMany
|
||||||
|
exam_answers: [ExamMemberRankAnswer]! @belongsToMany
|
||||||
|
}
|
||||||
|
type ExamTyperating {
|
||||||
|
id: ID!
|
||||||
|
member_id: ID!
|
||||||
|
typerating: Typerating! @belongsTo
|
||||||
|
start: DateTime!
|
||||||
|
status: ExamStatus!
|
||||||
|
exam_questions: [ExamTyperatingQuestion]! @belongsToMany
|
||||||
|
exam_answers: [ExamTyperatingAnswer]! @belongsToMany
|
||||||
|
}
|
||||||
|
type ExamMemberRankQuestion {
|
||||||
|
id: ID!
|
||||||
|
member_rank: MemberRank! @belongsTo
|
||||||
|
in_use: Boolean!
|
||||||
|
mandatory: Boolean!
|
||||||
|
text: String!
|
||||||
|
picture: String!
|
||||||
|
exam_answers: [ExamMemberRankAnswer]! @hasMany
|
||||||
|
exam_member_ranks: [ExamMemberRank]! @belongsToMany
|
||||||
|
}
|
||||||
|
type ExamTyperatingQuestion {
|
||||||
|
id: ID!
|
||||||
|
typerating: Typerating! @belongsTo
|
||||||
|
in_use: Boolean!
|
||||||
|
mandatory: Boolean!
|
||||||
|
text: String!
|
||||||
|
picture: String!
|
||||||
|
exam_answers: [ExamTyperatingAnswer]! @hasMany
|
||||||
|
exma_typeratings: [ExamTyperating]! @belongsToMany
|
||||||
|
}
|
||||||
|
type ExamMemberRankAnswer {
|
||||||
|
id: ID!
|
||||||
|
exam_question: ExamMemberRankQuestion! @belongsTo
|
||||||
|
text: String!
|
||||||
|
is_correct: Boolean!
|
||||||
|
exam_member_ranks: [ExamMemberRank]! @belongsToMany
|
||||||
|
}
|
||||||
|
type ExamTyperatingAnswer {
|
||||||
|
id: ID!
|
||||||
|
exam_question: ExamTyperatingQuestion! @belongsTo
|
||||||
|
text: String!
|
||||||
|
is_correct: Boolean!
|
||||||
|
exam_typeratings: [ExamTyperating]! @belongsToMany
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ExamStatus {
|
||||||
|
open
|
||||||
|
pending
|
||||||
|
closed
|
||||||
|
validated
|
||||||
|
}
|
||||||
|
|
||||||
|
scalar DateTime
|
||||||
|
@scalar(class: "Nuwave\\Lighthouse\\Schema\\Types\\Scalars\\DateTime")
|
||||||
@@ -1,16 +1,16 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
if(!function_exists("join_paths")) {
|
if (!function_exists('join_paths')) {
|
||||||
function join_paths()
|
function join_paths()
|
||||||
{
|
{
|
||||||
$paths = array();
|
$paths = [];
|
||||||
|
|
||||||
foreach (func_get_args() as $arg) {
|
foreach (func_get_args() as $arg) {
|
||||||
if ($arg !== '') {
|
if ($arg !== '') {
|
||||||
$paths[] = $arg;
|
$paths[] = $arg;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return preg_replace('#/+#', '/', join('/', $paths));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return preg_replace('#/+#', '/', join('/', $paths));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+32
-6
@@ -1,9 +1,35 @@
|
|||||||
<?php return [
|
<?php return [
|
||||||
'plugin' => [
|
'plugin' => [
|
||||||
'name' => 'GraphQL',
|
'name' => 'GA GraphQL',
|
||||||
'description' => '',
|
'description' => '',
|
||||||
|
],
|
||||||
|
'menu' => [
|
||||||
|
'main' => 'GA GraphQL',
|
||||||
|
'keys' => 'GraphQL Keys',
|
||||||
|
'playground' => 'GraphiQL',
|
||||||
|
],
|
||||||
|
'permission' => [
|
||||||
|
'tab' => [
|
||||||
|
'schemas' => 'German Airlines VA - GraphQL',
|
||||||
],
|
],
|
||||||
'menu' => [
|
'label' => [
|
||||||
'main' => 'GraphQL',
|
'schemas' => 'Access GraphiQL',
|
||||||
],
|
],
|
||||||
];
|
],
|
||||||
|
'settings' => [
|
||||||
|
'label' => 'German Airlines VA - GraphQL',
|
||||||
|
'description' => 'Configure GrapQL',
|
||||||
|
'general' => 'General',
|
||||||
|
'engine' => 'Engine',
|
||||||
|
'enable_cache' => [
|
||||||
|
'label' => 'Enable Schema Cache',
|
||||||
|
'comment' =>
|
||||||
|
'A large part of the Schema generation is parsing the various graph definition into an AST. These operations are pretty expensive so it is recommended to enable caching in production mode.',
|
||||||
|
],
|
||||||
|
'batched_queries' => [
|
||||||
|
'label' => 'Batched Queries',
|
||||||
|
'comment' =>
|
||||||
|
'GraphQL query batching means sending multiple queries to the server in one request. You may set this flag to process/deny batched queries.',
|
||||||
|
],
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\Graphql\Models;
|
||||||
|
|
||||||
|
use Model;
|
||||||
|
|
||||||
|
class GraphqlKey extends Model
|
||||||
|
{
|
||||||
|
use \October\Rain\Database\Traits\Validation;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Disable timestamps by default.
|
||||||
|
* Remove this line if timestamps are defined in the database table.
|
||||||
|
*/
|
||||||
|
public $timestamps = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string The database table used by the model.
|
||||||
|
*/
|
||||||
|
public $table = 'graphql_keys';
|
||||||
|
protected $connection = 'germanairlinesva_graphql';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var array Validation rules
|
||||||
|
*/
|
||||||
|
public $rules = [
|
||||||
|
'member_id' => 'required',
|
||||||
|
'key' => 'required',
|
||||||
|
];
|
||||||
|
|
||||||
|
public $belongsTo = [
|
||||||
|
'member' => 'GermanAirlinesVa\Social\Models\Member',
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\Graphql\Models;
|
||||||
|
|
||||||
|
use Model;
|
||||||
|
|
||||||
|
class Settings extends Model
|
||||||
|
{
|
||||||
|
public $implement = ['System.Behaviors.SettingsModel'];
|
||||||
|
|
||||||
|
public $settingsCode = 'germanairlinesva_graphql_settings';
|
||||||
|
|
||||||
|
public $settingsFields = 'fields.yaml';
|
||||||
|
}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
columns:
|
||||||
|
id:
|
||||||
|
label: id
|
||||||
|
type: number
|
||||||
|
member:
|
||||||
|
label: member
|
||||||
|
type: text
|
||||||
|
relation: member
|
||||||
|
valueFrom: name
|
||||||
|
valid_from:
|
||||||
|
label: valid_from
|
||||||
|
type: datetime
|
||||||
|
key:
|
||||||
|
label: key
|
||||||
|
type: text
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
tabs:
|
||||||
|
fields:
|
||||||
|
enable_cache:
|
||||||
|
label: 'germanairlinesva.graphql::lang.settings.enable_cache.label'
|
||||||
|
span: left
|
||||||
|
default: false
|
||||||
|
type: checkbox
|
||||||
|
comment: 'germanairlinesva.graphql::lang.settings.enable_cache.comment'
|
||||||
|
tab: 'germanairlinesva.graphql::lang.settings.engine'
|
||||||
|
batched_queries:
|
||||||
|
label: 'germanairlinesva.graphql::lang.settings.batched_queries.label'
|
||||||
|
span: right
|
||||||
|
default: true
|
||||||
|
type: checkbox
|
||||||
|
comment: 'germanairlinesva.graphql::lang.settings.batched_queries.comment'
|
||||||
|
tab: 'germanairlinesva.graphql::lang.settings.engine'
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
{
|
||||||
|
"name": "graphql",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"main": "Plugin.php",
|
||||||
|
"repository": "https://git.hofmannnet.myhome-server.de/GermanAirlines/GermanAirlinesVA-GraphQL.git",
|
||||||
|
"author": "German Airlines VA",
|
||||||
|
"license": "MIT",
|
||||||
|
"devDependencies": {
|
||||||
|
"@prettier/plugin-php": "^0.17.3",
|
||||||
|
"prettier": "^2.3.2"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"preinstall": "git config core.hooksPath .hooks",
|
||||||
|
"format": "prettier --write './**/*.{php,html,yaml}'"
|
||||||
|
}
|
||||||
|
}
|
||||||
+25
-5
@@ -1,6 +1,26 @@
|
|||||||
plugin:
|
plugin:
|
||||||
name: 'germanairlinesva.graphql::lang.plugin.name'
|
name: 'germanairlinesva.graphql::lang.plugin.name'
|
||||||
description: 'germanairlinesva.graphql::lang.plugin.description'
|
description: 'germanairlinesva.graphql::lang.plugin.description'
|
||||||
author: 'German Airlines VA'
|
author: 'German Airlines VA'
|
||||||
icon: oc-icon-database
|
icon: oc-icon-database
|
||||||
homepage: ''
|
homepage: ''
|
||||||
|
permissions:
|
||||||
|
germanairlinesva.graphql.master:
|
||||||
|
tab: 'germanairlinesva.graphql::lang.permission.tab.schemas'
|
||||||
|
label: 'germanairlinesva.graphql::lang.permission.label.schemas'
|
||||||
|
navigation:
|
||||||
|
menu:
|
||||||
|
label: 'germanairlinesva.graphql::lang.menu.main'
|
||||||
|
url: /
|
||||||
|
icon: icon-database
|
||||||
|
permissions:
|
||||||
|
- germanairlinesva.graphql.master
|
||||||
|
sideMenu:
|
||||||
|
side-menu-item:
|
||||||
|
label: 'germanairlinesva.graphql::lang.menu.playground'
|
||||||
|
url: germanairlinesva/graphql/playground
|
||||||
|
icon: icon-database
|
||||||
|
side-menu-item2:
|
||||||
|
label: 'germanairlinesva.graphql::lang.menu.keys'
|
||||||
|
url: germanairlinesva/graphql/graphqlkey
|
||||||
|
icon: icon-key
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\GraphQl\Updates;
|
||||||
|
|
||||||
|
use Schema;
|
||||||
|
use October\Rain\Database\Updates\Migration;
|
||||||
|
|
||||||
|
class BuilderTableCreateGermanAirlinesVaGraphQlDeferredBindings extends Migration
|
||||||
|
{
|
||||||
|
public function up()
|
||||||
|
{
|
||||||
|
Schema::connection('germanairlinesva_graphql')->create('deferred_bindings', function ($table) {
|
||||||
|
$table->engine = 'InnoDB';
|
||||||
|
$table->increments('id')->unsigned();
|
||||||
|
$table->string('master_type');
|
||||||
|
$table->string('master_field');
|
||||||
|
$table->string('slave_type');
|
||||||
|
$table->integer('slave_id');
|
||||||
|
$table->mediumText('pivot_data')->nullable();
|
||||||
|
$table->string('session_key');
|
||||||
|
$table->boolean('is_bind')->default(true);
|
||||||
|
$table->timestamps();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down()
|
||||||
|
{
|
||||||
|
Schema::connection('germanairlinesva_graphql')->disableForeignKeyConstraints();
|
||||||
|
Schema::connection('germanairlinesva_graphql')->dropIfExists('deferred_bindings');
|
||||||
|
Schema::connection('germanairlinesva_graphql')->enableForeignKeyConstraints();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
<?php namespace GermanAirlinesVa\Graphql\Updates;
|
||||||
|
|
||||||
|
use DB;
|
||||||
|
use Schema;
|
||||||
|
use October\Rain\Database\Updates\Migration;
|
||||||
|
|
||||||
|
class BuilderTableCreateGermanAirlinesVaGraphqlGraphqlKeys extends Migration
|
||||||
|
{
|
||||||
|
public function up()
|
||||||
|
{
|
||||||
|
Schema::connection('germanairlinesva_graphql')->create('graphql_keys', function ($table) {
|
||||||
|
$table->engine = 'InnoDB';
|
||||||
|
$table->bigIncrements('id')->unsigned();
|
||||||
|
$table->bigInteger('member_id')->unsigned();
|
||||||
|
$table->datetime('valid_from')->default(DB::raw('NOW()'));
|
||||||
|
$table->string('key');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down()
|
||||||
|
{
|
||||||
|
Schema::connection('germanairlinesva_graphql')->disableForeignKeyConstraints();
|
||||||
|
Schema::connection('germanairlinesva_graphql')->dropIfExists('graphql_keys');
|
||||||
|
Schema::connection('germanairlinesva_graphql')->enableForeignKeyConstraints();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,2 +1,7 @@
|
|||||||
1.0.1:
|
1.0.1:
|
||||||
- Initialize plugin.
|
- 'Initialize plugin.'
|
||||||
|
- 'Create table deferred_bindings'
|
||||||
|
- builder_table_create_deferred_bindings.php
|
||||||
|
1.0.2:
|
||||||
|
- 'Create table graphql_keys'
|
||||||
|
- builder_table_create_graphql_keys.php
|
||||||
|
|||||||
Vendored
+40
-4
@@ -37,11 +37,13 @@ namespace Composer\Autoload;
|
|||||||
*
|
*
|
||||||
* @author Fabien Potencier <fabien@symfony.com>
|
* @author Fabien Potencier <fabien@symfony.com>
|
||||||
* @author Jordi Boggiano <j.boggiano@seld.be>
|
* @author Jordi Boggiano <j.boggiano@seld.be>
|
||||||
* @see http://www.php-fig.org/psr/psr-0/
|
* @see https://www.php-fig.org/psr/psr-0/
|
||||||
* @see http://www.php-fig.org/psr/psr-4/
|
* @see https://www.php-fig.org/psr/psr-4/
|
||||||
*/
|
*/
|
||||||
class ClassLoader
|
class ClassLoader
|
||||||
{
|
{
|
||||||
|
private $vendorDir;
|
||||||
|
|
||||||
// PSR-4
|
// PSR-4
|
||||||
private $prefixLengthsPsr4 = array();
|
private $prefixLengthsPsr4 = array();
|
||||||
private $prefixDirsPsr4 = array();
|
private $prefixDirsPsr4 = array();
|
||||||
@@ -57,10 +59,17 @@ class ClassLoader
|
|||||||
private $missingClasses = array();
|
private $missingClasses = array();
|
||||||
private $apcuPrefix;
|
private $apcuPrefix;
|
||||||
|
|
||||||
|
private static $registeredLoaders = array();
|
||||||
|
|
||||||
|
public function __construct($vendorDir = null)
|
||||||
|
{
|
||||||
|
$this->vendorDir = $vendorDir;
|
||||||
|
}
|
||||||
|
|
||||||
public function getPrefixes()
|
public function getPrefixes()
|
||||||
{
|
{
|
||||||
if (!empty($this->prefixesPsr0)) {
|
if (!empty($this->prefixesPsr0)) {
|
||||||
return call_user_func_array('array_merge', $this->prefixesPsr0);
|
return call_user_func_array('array_merge', array_values($this->prefixesPsr0));
|
||||||
}
|
}
|
||||||
|
|
||||||
return array();
|
return array();
|
||||||
@@ -300,6 +309,17 @@ class ClassLoader
|
|||||||
public function register($prepend = false)
|
public function register($prepend = false)
|
||||||
{
|
{
|
||||||
spl_autoload_register(array($this, 'loadClass'), true, $prepend);
|
spl_autoload_register(array($this, 'loadClass'), true, $prepend);
|
||||||
|
|
||||||
|
if (null === $this->vendorDir) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($prepend) {
|
||||||
|
self::$registeredLoaders = array($this->vendorDir => $this) + self::$registeredLoaders;
|
||||||
|
} else {
|
||||||
|
unset(self::$registeredLoaders[$this->vendorDir]);
|
||||||
|
self::$registeredLoaders[$this->vendorDir] = $this;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -308,13 +328,17 @@ class ClassLoader
|
|||||||
public function unregister()
|
public function unregister()
|
||||||
{
|
{
|
||||||
spl_autoload_unregister(array($this, 'loadClass'));
|
spl_autoload_unregister(array($this, 'loadClass'));
|
||||||
|
|
||||||
|
if (null !== $this->vendorDir) {
|
||||||
|
unset(self::$registeredLoaders[$this->vendorDir]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loads the given class or interface.
|
* Loads the given class or interface.
|
||||||
*
|
*
|
||||||
* @param string $class The name of the class
|
* @param string $class The name of the class
|
||||||
* @return bool|null True if loaded, null otherwise
|
* @return true|null True if loaded, null otherwise
|
||||||
*/
|
*/
|
||||||
public function loadClass($class)
|
public function loadClass($class)
|
||||||
{
|
{
|
||||||
@@ -323,6 +347,8 @@ class ClassLoader
|
|||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -367,6 +393,16 @@ class ClassLoader
|
|||||||
return $file;
|
return $file;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the currently registered loaders indexed by their corresponding vendor directories.
|
||||||
|
*
|
||||||
|
* @return self[]
|
||||||
|
*/
|
||||||
|
public static function getRegisteredLoaders()
|
||||||
|
{
|
||||||
|
return self::$registeredLoaders;
|
||||||
|
}
|
||||||
|
|
||||||
private function findFileWithExtension($class, $ext)
|
private function findFileWithExtension($class, $ext)
|
||||||
{
|
{
|
||||||
// PSR-4 lookup
|
// PSR-4 lookup
|
||||||
|
|||||||
+2
-7
@@ -6,12 +6,7 @@ $vendorDir = dirname(dirname(__FILE__));
|
|||||||
$baseDir = dirname($vendorDir);
|
$baseDir = dirname($vendorDir);
|
||||||
|
|
||||||
return array(
|
return array(
|
||||||
'ArithmeticError' => $vendorDir . '/symfony/polyfill-php70/Resources/stubs/ArithmeticError.php',
|
'Composer\\InstalledVersions' => $vendorDir . '/composer/InstalledVersions.php',
|
||||||
'AssertionError' => $vendorDir . '/symfony/polyfill-php70/Resources/stubs/AssertionError.php',
|
'JsonException' => $vendorDir . '/symfony/polyfill-php73/Resources/stubs/JsonException.php',
|
||||||
'DivisionByZeroError' => $vendorDir . '/symfony/polyfill-php70/Resources/stubs/DivisionByZeroError.php',
|
|
||||||
'Error' => $vendorDir . '/symfony/polyfill-php70/Resources/stubs/Error.php',
|
|
||||||
'Normalizer' => $vendorDir . '/symfony/polyfill-intl-normalizer/Resources/stubs/Normalizer.php',
|
'Normalizer' => $vendorDir . '/symfony/polyfill-intl-normalizer/Resources/stubs/Normalizer.php',
|
||||||
'ParseError' => $vendorDir . '/symfony/polyfill-php70/Resources/stubs/ParseError.php',
|
|
||||||
'SessionUpdateTimestampHandlerInterface' => $vendorDir . '/symfony/polyfill-php70/Resources/stubs/SessionUpdateTimestampHandlerInterface.php',
|
|
||||||
'TypeError' => $vendorDir . '/symfony/polyfill-php70/Resources/stubs/TypeError.php',
|
|
||||||
);
|
);
|
||||||
|
|||||||
Vendored
+91
-5
@@ -6,12 +6,98 @@ $vendorDir = dirname(dirname(__FILE__));
|
|||||||
$baseDir = dirname($vendorDir);
|
$baseDir = dirname($vendorDir);
|
||||||
|
|
||||||
return array(
|
return array(
|
||||||
'7b11c4dc42b3b3023073cb14e519683c' => $vendorDir . '/ralouphie/getallheaders/src/getallheaders.php',
|
'51fcf4e06c07cc00c920b44bcd900e7a' => $vendorDir . '/thecodingmachine/safe/deprecated/apc.php',
|
||||||
|
'47f619d9197b36cf5ab70738d7743fe2' => $vendorDir . '/thecodingmachine/safe/deprecated/libevent.php',
|
||||||
|
'ea6bb8a12ef9b68f6ada99058e530760' => $vendorDir . '/thecodingmachine/safe/deprecated/mssql.php',
|
||||||
|
'9a29089eb3ce41a446744c68a00f118c' => $vendorDir . '/thecodingmachine/safe/deprecated/stats.php',
|
||||||
|
'72243e5536b63e298acb6476f01f1aff' => $vendorDir . '/thecodingmachine/safe/lib/special_cases.php',
|
||||||
|
'3f648889e687f31c52f949ba8a9d0873' => $vendorDir . '/thecodingmachine/safe/generated/apache.php',
|
||||||
|
'eeb4581d958421a4244aaa4167c6a575' => $vendorDir . '/thecodingmachine/safe/generated/apcu.php',
|
||||||
|
'04cb0b3c1dac5b5ddb23c14e3d66dbe9' => $vendorDir . '/thecodingmachine/safe/generated/array.php',
|
||||||
|
'450b332a74a9a21e043c5e953485a791' => $vendorDir . '/thecodingmachine/safe/generated/bzip2.php',
|
||||||
|
'6e9b7954ecfd7cbb9ca239319d1acdb6' => $vendorDir . '/thecodingmachine/safe/generated/calendar.php',
|
||||||
|
'2c6d7e8bd2de9a272a9d4d43b0a4304a' => $vendorDir . '/thecodingmachine/safe/generated/classobj.php',
|
||||||
|
'0b8231c1ad0865447c988a4c16b4001f' => $vendorDir . '/thecodingmachine/safe/generated/com.php',
|
||||||
|
'7643a71fe1c3256058c8fee234cb86e5' => $vendorDir . '/thecodingmachine/safe/generated/cubrid.php',
|
||||||
|
'68e1365710575942efc1d55000032cee' => $vendorDir . '/thecodingmachine/safe/generated/curl.php',
|
||||||
|
'02fd26bca803106c5b942a7197c3ad8b' => $vendorDir . '/thecodingmachine/safe/generated/datetime.php',
|
||||||
|
'f4817dcbd956cd221b1c31f6fbd5749c' => $vendorDir . '/thecodingmachine/safe/generated/dir.php',
|
||||||
|
'51c3f2d10ca61a70dbcea0e38d8e902d' => $vendorDir . '/thecodingmachine/safe/generated/eio.php',
|
||||||
|
'1d34f34327ca3e81535963016e3be2c3' => $vendorDir . '/thecodingmachine/safe/generated/errorfunc.php',
|
||||||
|
'4fd0ba2d3717b0424d474bebfdafa2b4' => $vendorDir . '/thecodingmachine/safe/generated/exec.php',
|
||||||
|
'98f4dae054bc7fb19c13be14935cbdd3' => $vendorDir . '/thecodingmachine/safe/generated/fileinfo.php',
|
||||||
|
'5530ae063ba88323eaf0a07904efdf85' => $vendorDir . '/thecodingmachine/safe/generated/filesystem.php',
|
||||||
|
'633f4f134975d70e97bddad83348e91a' => $vendorDir . '/thecodingmachine/safe/generated/filter.php',
|
||||||
|
'fbd163fc68c5faf73d5ed4002ffd836d' => $vendorDir . '/thecodingmachine/safe/generated/fpm.php',
|
||||||
|
'21b511999d61411fab0692ff8795bbed' => $vendorDir . '/thecodingmachine/safe/generated/ftp.php',
|
||||||
|
'85fbd73fc92365cd90526b0ea03cae3a' => $vendorDir . '/thecodingmachine/safe/generated/funchand.php',
|
||||||
|
'51df9c146e0b7dcbdf358d8abd24dbdc' => $vendorDir . '/thecodingmachine/safe/generated/gmp.php',
|
||||||
|
'93bb7fe678d7dcfb1322f8e3475a48b0' => $vendorDir . '/thecodingmachine/safe/generated/gnupg.php',
|
||||||
|
'c171ba99cf316379ff66468392bf4950' => $vendorDir . '/thecodingmachine/safe/generated/hash.php',
|
||||||
|
'5ab4aad4c28e468209fbfcceb2e5e6a5' => $vendorDir . '/thecodingmachine/safe/generated/ibase.php',
|
||||||
|
'4d57409c5e8e576b0c64c08d9d731cfb' => $vendorDir . '/thecodingmachine/safe/generated/ibmDb2.php',
|
||||||
|
'eeb246d5403972a9d62106e4a4883496' => $vendorDir . '/thecodingmachine/safe/generated/iconv.php',
|
||||||
|
'c28a05f498c01b810a714f7214b7a8da' => $vendorDir . '/thecodingmachine/safe/generated/image.php',
|
||||||
|
'8063cd92acdf00fd978b5599eb7cc142' => $vendorDir . '/thecodingmachine/safe/generated/imap.php',
|
||||||
|
'8bd26dbe768e9c9599edad7b198e5446' => $vendorDir . '/thecodingmachine/safe/generated/info.php',
|
||||||
|
'0c577fe603b029d4b65c84376b15dbd5' => $vendorDir . '/thecodingmachine/safe/generated/ingres-ii.php',
|
||||||
|
'd4362910bde43c0f956b52527effd7d4' => $vendorDir . '/thecodingmachine/safe/generated/inotify.php',
|
||||||
|
'696ba49197d9b55f0428a12bb5a818e1' => $vendorDir . '/thecodingmachine/safe/generated/json.php',
|
||||||
|
'9818aaa99c8647c63f8ef62b7a368160' => $vendorDir . '/thecodingmachine/safe/generated/ldap.php',
|
||||||
|
'bcf523ff2a195eb08e0fbb668ed784d0' => $vendorDir . '/thecodingmachine/safe/generated/libxml.php',
|
||||||
|
'68be68a9a8b95bb56cab6109ff03bc88' => $vendorDir . '/thecodingmachine/safe/generated/lzf.php',
|
||||||
|
'bdca804bb0904ea9f53f328dfc0bb8a5' => $vendorDir . '/thecodingmachine/safe/generated/mailparse.php',
|
||||||
|
'b0a3fcac3eaf55445796d6af26b89366' => $vendorDir . '/thecodingmachine/safe/generated/mbstring.php',
|
||||||
|
'98de16b8db03eb0cb4d318b4402215a6' => $vendorDir . '/thecodingmachine/safe/generated/misc.php',
|
||||||
|
'c112440003b56e243b192c11fa9d836e' => $vendorDir . '/thecodingmachine/safe/generated/msql.php',
|
||||||
|
'7cefd81607cd21b8b3a15656eb6465f5' => $vendorDir . '/thecodingmachine/safe/generated/mysql.php',
|
||||||
|
'aaf438b080089c6d0686679cd34aa72e' => $vendorDir . '/thecodingmachine/safe/generated/mysqli.php',
|
||||||
|
'df0ef890e9afbf95f3924feb1c7a89f3' => $vendorDir . '/thecodingmachine/safe/generated/mysqlndMs.php',
|
||||||
|
'db595fee5972867e45c5327010d78735' => $vendorDir . '/thecodingmachine/safe/generated/mysqlndQc.php',
|
||||||
|
'cbac956836b72483dcff1ac39d5c0a0f' => $vendorDir . '/thecodingmachine/safe/generated/network.php',
|
||||||
|
'6c8f89dfbdc117d7871f572269363f25' => $vendorDir . '/thecodingmachine/safe/generated/oci8.php',
|
||||||
|
'169a669966a45c06bf55ed029122729b' => $vendorDir . '/thecodingmachine/safe/generated/opcache.php',
|
||||||
|
'def61bf4fecd4d4bca7354919cd69302' => $vendorDir . '/thecodingmachine/safe/generated/openssl.php',
|
||||||
|
'26bb010649a6d32d4120181458aa6ef2' => $vendorDir . '/thecodingmachine/safe/generated/outcontrol.php',
|
||||||
|
'1212c201fe43c7492a085b2c71505e0f' => $vendorDir . '/thecodingmachine/safe/generated/password.php',
|
||||||
|
'002ebcb842e2c0d5b7f67fe64cc93158' => $vendorDir . '/thecodingmachine/safe/generated/pcntl.php',
|
||||||
|
'86df38612982dade72c7085ce7eca81f' => $vendorDir . '/thecodingmachine/safe/generated/pcre.php',
|
||||||
|
'1cacc3e65f82a473fbd5507c7ce4385d' => $vendorDir . '/thecodingmachine/safe/generated/pdf.php',
|
||||||
|
'1fc22f445c69ea8706e82fce301c0831' => $vendorDir . '/thecodingmachine/safe/generated/pgsql.php',
|
||||||
|
'c70b42561584f7144bff38cd63c4eef3' => $vendorDir . '/thecodingmachine/safe/generated/posix.php',
|
||||||
|
'9923214639c32ca5173db03a177d3b63' => $vendorDir . '/thecodingmachine/safe/generated/ps.php',
|
||||||
|
'7e9c3f8eae2b5bf42205c4f1295cb7a7' => $vendorDir . '/thecodingmachine/safe/generated/pspell.php',
|
||||||
|
'91aa91f6245c349c2e2e88bd0025f199' => $vendorDir . '/thecodingmachine/safe/generated/readline.php',
|
||||||
|
'd43773cacb9e5e8e897aa255e32007d1' => $vendorDir . '/thecodingmachine/safe/generated/rpminfo.php',
|
||||||
|
'f053a3849e9e8383762b34b91db0320b' => $vendorDir . '/thecodingmachine/safe/generated/rrd.php',
|
||||||
|
'775b964f72f827a1bf87c65ab5b10800' => $vendorDir . '/thecodingmachine/safe/generated/sem.php',
|
||||||
|
'816428bd69c29ab5e1ed622af5dca0cd' => $vendorDir . '/thecodingmachine/safe/generated/session.php',
|
||||||
|
'5093e233bedbefaef0df262bfbab0a5c' => $vendorDir . '/thecodingmachine/safe/generated/shmop.php',
|
||||||
|
'01352920b0151f17e671266e44b52536' => $vendorDir . '/thecodingmachine/safe/generated/simplexml.php',
|
||||||
|
'b080617b1d949683c2e37f8f01dc0e15' => $vendorDir . '/thecodingmachine/safe/generated/sockets.php',
|
||||||
|
'2708aa182ddcfe6ce27c96acaaa40f69' => $vendorDir . '/thecodingmachine/safe/generated/sodium.php',
|
||||||
|
'f1b96cb260a5baeea9a7285cda82a1ec' => $vendorDir . '/thecodingmachine/safe/generated/solr.php',
|
||||||
|
'3fd8853757d0fe3557c179efb807afeb' => $vendorDir . '/thecodingmachine/safe/generated/spl.php',
|
||||||
|
'9312ce96a51c846913fcda5f186d58dd' => $vendorDir . '/thecodingmachine/safe/generated/sqlsrv.php',
|
||||||
|
'd3eb383ad0b8b962b29dc4afd29d6715' => $vendorDir . '/thecodingmachine/safe/generated/ssdeep.php',
|
||||||
|
'42a09bc448f441a0b9f9367ea975c0bf' => $vendorDir . '/thecodingmachine/safe/generated/ssh2.php',
|
||||||
|
'ef711077d356d1b33ca0b10b67b0be8f' => $vendorDir . '/thecodingmachine/safe/generated/stream.php',
|
||||||
|
'764b09f6df081cbb2807b97c6ace3866' => $vendorDir . '/thecodingmachine/safe/generated/strings.php',
|
||||||
|
'ef241678769fee4a44aaa288f3b78aa1' => $vendorDir . '/thecodingmachine/safe/generated/swoole.php',
|
||||||
|
'0efc8f6778cba932b9e2a89e28de2452' => $vendorDir . '/thecodingmachine/safe/generated/uodbc.php',
|
||||||
|
'd383d32907b98af53ee9208c62204fd0' => $vendorDir . '/thecodingmachine/safe/generated/uopz.php',
|
||||||
|
'2fd2e4060f7fe772660f002ce38f0b71' => $vendorDir . '/thecodingmachine/safe/generated/url.php',
|
||||||
|
'782249e03deebeaf57b9991ff5493aa0' => $vendorDir . '/thecodingmachine/safe/generated/var.php',
|
||||||
|
'344440cd1cd7200fdb4f12af0d3c587f' => $vendorDir . '/thecodingmachine/safe/generated/xdiff.php',
|
||||||
|
'3599f369219c658a5fb6c4fe66832f62' => $vendorDir . '/thecodingmachine/safe/generated/xml.php',
|
||||||
|
'7fcd313da9fae337051b091b3492c21b' => $vendorDir . '/thecodingmachine/safe/generated/xmlrpc.php',
|
||||||
|
'd668c74cfa92d893b582356733d9a80e' => $vendorDir . '/thecodingmachine/safe/generated/yaml.php',
|
||||||
|
'4af1dca6db8c527c6eed27bff85ff0e5' => $vendorDir . '/thecodingmachine/safe/generated/yaz.php',
|
||||||
|
'fe43ca06499ac37bc2dedd823af71eb5' => $vendorDir . '/thecodingmachine/safe/generated/zip.php',
|
||||||
|
'356736db98a6834f0a886b8d509b0ecd' => $vendorDir . '/thecodingmachine/safe/generated/zlib.php',
|
||||||
'e69f7f6ee287b969198c3c9d6777bd38' => $vendorDir . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
'e69f7f6ee287b969198c3c9d6777bd38' => $vendorDir . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
||||||
'023d27dca8066ef29e6739335ea73bad' => $vendorDir . '/symfony/polyfill-php70/bootstrap.php',
|
|
||||||
'25072dd6e2470089de65ae7bf11d3109' => $vendorDir . '/symfony/polyfill-php72/bootstrap.php',
|
'25072dd6e2470089de65ae7bf11d3109' => $vendorDir . '/symfony/polyfill-php72/bootstrap.php',
|
||||||
'c964ee0ededf28c96ebd9db5099ef910' => $vendorDir . '/guzzlehttp/promises/src/functions_include.php',
|
'8825ede83f2f289127722d4e842cf7e8' => $vendorDir . '/symfony/polyfill-intl-grapheme/bootstrap.php',
|
||||||
'a0edc8309cc5e1d60e3047b5df6b7052' => $vendorDir . '/guzzlehttp/psr7/src/functions_include.php',
|
|
||||||
'f598d06aa772fa33d905e87be6398fb1' => $vendorDir . '/symfony/polyfill-intl-idn/bootstrap.php',
|
'f598d06aa772fa33d905e87be6398fb1' => $vendorDir . '/symfony/polyfill-intl-idn/bootstrap.php',
|
||||||
'37a3dc5111fe8f707ab4c132ef1dbc62' => $vendorDir . '/guzzlehttp/guzzle/src/functions_include.php',
|
'0d59ee240a4cd96ddbb4ff164fccea4d' => $vendorDir . '/symfony/polyfill-php73/bootstrap.php',
|
||||||
);
|
);
|
||||||
|
|||||||
Vendored
+10
-5
@@ -6,14 +6,19 @@ $vendorDir = dirname(dirname(__FILE__));
|
|||||||
$baseDir = dirname($vendorDir);
|
$baseDir = dirname($vendorDir);
|
||||||
|
|
||||||
return array(
|
return array(
|
||||||
|
'Symfony\\Polyfill\\Php73\\' => array($vendorDir . '/symfony/polyfill-php73'),
|
||||||
'Symfony\\Polyfill\\Php72\\' => array($vendorDir . '/symfony/polyfill-php72'),
|
'Symfony\\Polyfill\\Php72\\' => array($vendorDir . '/symfony/polyfill-php72'),
|
||||||
'Symfony\\Polyfill\\Php70\\' => array($vendorDir . '/symfony/polyfill-php70'),
|
|
||||||
'Symfony\\Polyfill\\Intl\\Normalizer\\' => array($vendorDir . '/symfony/polyfill-intl-normalizer'),
|
'Symfony\\Polyfill\\Intl\\Normalizer\\' => array($vendorDir . '/symfony/polyfill-intl-normalizer'),
|
||||||
'Symfony\\Polyfill\\Intl\\Idn\\' => array($vendorDir . '/symfony/polyfill-intl-idn'),
|
'Symfony\\Polyfill\\Intl\\Idn\\' => array($vendorDir . '/symfony/polyfill-intl-idn'),
|
||||||
'Psr\\Http\\Message\\' => array($vendorDir . '/psr/http-message/src'),
|
'Symfony\\Polyfill\\Intl\\Grapheme\\' => array($vendorDir . '/symfony/polyfill-intl-grapheme'),
|
||||||
|
'Safe\\' => array($vendorDir . '/thecodingmachine/safe/lib', $vendorDir . '/thecodingmachine/safe/deprecated', $vendorDir . '/thecodingmachine/safe/generated'),
|
||||||
|
'Psr\\SimpleCache\\' => array($vendorDir . '/psr/simple-cache/src'),
|
||||||
|
'Psr\\Log\\' => array($vendorDir . '/psr/log/Psr/Log'),
|
||||||
|
'Psr\\EventDispatcher\\' => array($vendorDir . '/psr/event-dispatcher/src'),
|
||||||
|
'Psr\\Container\\' => array($vendorDir . '/psr/container/src'),
|
||||||
'Nuwave\\Lighthouse\\' => array($vendorDir . '/nuwave/lighthouse/src'),
|
'Nuwave\\Lighthouse\\' => array($vendorDir . '/nuwave/lighthouse/src'),
|
||||||
'GuzzleHttp\\Psr7\\' => array($vendorDir . '/guzzlehttp/psr7/src'),
|
'Laragraph\\Utils\\' => array($vendorDir . '/laragraph/utils/src'),
|
||||||
'GuzzleHttp\\Promise\\' => array($vendorDir . '/guzzlehttp/promises/src'),
|
'HaydenPierce\\ClassFinder\\UnitTest\\' => array($vendorDir . '/haydenpierce/class-finder/test/unit'),
|
||||||
'GuzzleHttp\\' => array($vendorDir . '/guzzlehttp/guzzle/src'),
|
'HaydenPierce\\ClassFinder\\' => array($vendorDir . '/haydenpierce/class-finder/src'),
|
||||||
'GraphQL\\' => array($vendorDir . '/webonyx/graphql-php/src'),
|
'GraphQL\\' => array($vendorDir . '/webonyx/graphql-php/src'),
|
||||||
);
|
);
|
||||||
|
|||||||
Vendored
+7
-2
@@ -13,19 +13,24 @@ class ComposerAutoloaderInit02b791b67b928853969b061eb6816088
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return \Composer\Autoload\ClassLoader
|
||||||
|
*/
|
||||||
public static function getLoader()
|
public static function getLoader()
|
||||||
{
|
{
|
||||||
if (null !== self::$loader) {
|
if (null !== self::$loader) {
|
||||||
return self::$loader;
|
return self::$loader;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
require __DIR__ . '/platform_check.php';
|
||||||
|
|
||||||
spl_autoload_register(array('ComposerAutoloaderInit02b791b67b928853969b061eb6816088', 'loadClassLoader'), true, true);
|
spl_autoload_register(array('ComposerAutoloaderInit02b791b67b928853969b061eb6816088', 'loadClassLoader'), true, true);
|
||||||
self::$loader = $loader = new \Composer\Autoload\ClassLoader();
|
self::$loader = $loader = new \Composer\Autoload\ClassLoader(\dirname(\dirname(__FILE__)));
|
||||||
spl_autoload_unregister(array('ComposerAutoloaderInit02b791b67b928853969b061eb6816088', 'loadClassLoader'));
|
spl_autoload_unregister(array('ComposerAutoloaderInit02b791b67b928853969b061eb6816088', 'loadClassLoader'));
|
||||||
|
|
||||||
$useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
|
$useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
|
||||||
if ($useStaticLoader) {
|
if ($useStaticLoader) {
|
||||||
require_once __DIR__ . '/autoload_static.php';
|
require __DIR__ . '/autoload_static.php';
|
||||||
|
|
||||||
call_user_func(\Composer\Autoload\ComposerStaticInit02b791b67b928853969b061eb6816088::getInitializer($loader));
|
call_user_func(\Composer\Autoload\ComposerStaticInit02b791b67b928853969b061eb6816088::getInitializer($loader));
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
Vendored
+143
-29
@@ -7,50 +7,147 @@ namespace Composer\Autoload;
|
|||||||
class ComposerStaticInit02b791b67b928853969b061eb6816088
|
class ComposerStaticInit02b791b67b928853969b061eb6816088
|
||||||
{
|
{
|
||||||
public static $files = array (
|
public static $files = array (
|
||||||
'7b11c4dc42b3b3023073cb14e519683c' => __DIR__ . '/..' . '/ralouphie/getallheaders/src/getallheaders.php',
|
'51fcf4e06c07cc00c920b44bcd900e7a' => __DIR__ . '/..' . '/thecodingmachine/safe/deprecated/apc.php',
|
||||||
|
'47f619d9197b36cf5ab70738d7743fe2' => __DIR__ . '/..' . '/thecodingmachine/safe/deprecated/libevent.php',
|
||||||
|
'ea6bb8a12ef9b68f6ada99058e530760' => __DIR__ . '/..' . '/thecodingmachine/safe/deprecated/mssql.php',
|
||||||
|
'9a29089eb3ce41a446744c68a00f118c' => __DIR__ . '/..' . '/thecodingmachine/safe/deprecated/stats.php',
|
||||||
|
'72243e5536b63e298acb6476f01f1aff' => __DIR__ . '/..' . '/thecodingmachine/safe/lib/special_cases.php',
|
||||||
|
'3f648889e687f31c52f949ba8a9d0873' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/apache.php',
|
||||||
|
'eeb4581d958421a4244aaa4167c6a575' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/apcu.php',
|
||||||
|
'04cb0b3c1dac5b5ddb23c14e3d66dbe9' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/array.php',
|
||||||
|
'450b332a74a9a21e043c5e953485a791' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/bzip2.php',
|
||||||
|
'6e9b7954ecfd7cbb9ca239319d1acdb6' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/calendar.php',
|
||||||
|
'2c6d7e8bd2de9a272a9d4d43b0a4304a' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/classobj.php',
|
||||||
|
'0b8231c1ad0865447c988a4c16b4001f' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/com.php',
|
||||||
|
'7643a71fe1c3256058c8fee234cb86e5' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/cubrid.php',
|
||||||
|
'68e1365710575942efc1d55000032cee' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/curl.php',
|
||||||
|
'02fd26bca803106c5b942a7197c3ad8b' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/datetime.php',
|
||||||
|
'f4817dcbd956cd221b1c31f6fbd5749c' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/dir.php',
|
||||||
|
'51c3f2d10ca61a70dbcea0e38d8e902d' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/eio.php',
|
||||||
|
'1d34f34327ca3e81535963016e3be2c3' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/errorfunc.php',
|
||||||
|
'4fd0ba2d3717b0424d474bebfdafa2b4' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/exec.php',
|
||||||
|
'98f4dae054bc7fb19c13be14935cbdd3' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/fileinfo.php',
|
||||||
|
'5530ae063ba88323eaf0a07904efdf85' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/filesystem.php',
|
||||||
|
'633f4f134975d70e97bddad83348e91a' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/filter.php',
|
||||||
|
'fbd163fc68c5faf73d5ed4002ffd836d' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/fpm.php',
|
||||||
|
'21b511999d61411fab0692ff8795bbed' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ftp.php',
|
||||||
|
'85fbd73fc92365cd90526b0ea03cae3a' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/funchand.php',
|
||||||
|
'51df9c146e0b7dcbdf358d8abd24dbdc' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/gmp.php',
|
||||||
|
'93bb7fe678d7dcfb1322f8e3475a48b0' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/gnupg.php',
|
||||||
|
'c171ba99cf316379ff66468392bf4950' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/hash.php',
|
||||||
|
'5ab4aad4c28e468209fbfcceb2e5e6a5' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ibase.php',
|
||||||
|
'4d57409c5e8e576b0c64c08d9d731cfb' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ibmDb2.php',
|
||||||
|
'eeb246d5403972a9d62106e4a4883496' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/iconv.php',
|
||||||
|
'c28a05f498c01b810a714f7214b7a8da' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/image.php',
|
||||||
|
'8063cd92acdf00fd978b5599eb7cc142' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/imap.php',
|
||||||
|
'8bd26dbe768e9c9599edad7b198e5446' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/info.php',
|
||||||
|
'0c577fe603b029d4b65c84376b15dbd5' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ingres-ii.php',
|
||||||
|
'd4362910bde43c0f956b52527effd7d4' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/inotify.php',
|
||||||
|
'696ba49197d9b55f0428a12bb5a818e1' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/json.php',
|
||||||
|
'9818aaa99c8647c63f8ef62b7a368160' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ldap.php',
|
||||||
|
'bcf523ff2a195eb08e0fbb668ed784d0' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/libxml.php',
|
||||||
|
'68be68a9a8b95bb56cab6109ff03bc88' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/lzf.php',
|
||||||
|
'bdca804bb0904ea9f53f328dfc0bb8a5' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/mailparse.php',
|
||||||
|
'b0a3fcac3eaf55445796d6af26b89366' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/mbstring.php',
|
||||||
|
'98de16b8db03eb0cb4d318b4402215a6' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/misc.php',
|
||||||
|
'c112440003b56e243b192c11fa9d836e' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/msql.php',
|
||||||
|
'7cefd81607cd21b8b3a15656eb6465f5' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/mysql.php',
|
||||||
|
'aaf438b080089c6d0686679cd34aa72e' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/mysqli.php',
|
||||||
|
'df0ef890e9afbf95f3924feb1c7a89f3' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/mysqlndMs.php',
|
||||||
|
'db595fee5972867e45c5327010d78735' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/mysqlndQc.php',
|
||||||
|
'cbac956836b72483dcff1ac39d5c0a0f' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/network.php',
|
||||||
|
'6c8f89dfbdc117d7871f572269363f25' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/oci8.php',
|
||||||
|
'169a669966a45c06bf55ed029122729b' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/opcache.php',
|
||||||
|
'def61bf4fecd4d4bca7354919cd69302' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/openssl.php',
|
||||||
|
'26bb010649a6d32d4120181458aa6ef2' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/outcontrol.php',
|
||||||
|
'1212c201fe43c7492a085b2c71505e0f' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/password.php',
|
||||||
|
'002ebcb842e2c0d5b7f67fe64cc93158' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/pcntl.php',
|
||||||
|
'86df38612982dade72c7085ce7eca81f' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/pcre.php',
|
||||||
|
'1cacc3e65f82a473fbd5507c7ce4385d' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/pdf.php',
|
||||||
|
'1fc22f445c69ea8706e82fce301c0831' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/pgsql.php',
|
||||||
|
'c70b42561584f7144bff38cd63c4eef3' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/posix.php',
|
||||||
|
'9923214639c32ca5173db03a177d3b63' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ps.php',
|
||||||
|
'7e9c3f8eae2b5bf42205c4f1295cb7a7' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/pspell.php',
|
||||||
|
'91aa91f6245c349c2e2e88bd0025f199' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/readline.php',
|
||||||
|
'd43773cacb9e5e8e897aa255e32007d1' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/rpminfo.php',
|
||||||
|
'f053a3849e9e8383762b34b91db0320b' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/rrd.php',
|
||||||
|
'775b964f72f827a1bf87c65ab5b10800' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/sem.php',
|
||||||
|
'816428bd69c29ab5e1ed622af5dca0cd' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/session.php',
|
||||||
|
'5093e233bedbefaef0df262bfbab0a5c' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/shmop.php',
|
||||||
|
'01352920b0151f17e671266e44b52536' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/simplexml.php',
|
||||||
|
'b080617b1d949683c2e37f8f01dc0e15' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/sockets.php',
|
||||||
|
'2708aa182ddcfe6ce27c96acaaa40f69' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/sodium.php',
|
||||||
|
'f1b96cb260a5baeea9a7285cda82a1ec' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/solr.php',
|
||||||
|
'3fd8853757d0fe3557c179efb807afeb' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/spl.php',
|
||||||
|
'9312ce96a51c846913fcda5f186d58dd' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/sqlsrv.php',
|
||||||
|
'd3eb383ad0b8b962b29dc4afd29d6715' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ssdeep.php',
|
||||||
|
'42a09bc448f441a0b9f9367ea975c0bf' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/ssh2.php',
|
||||||
|
'ef711077d356d1b33ca0b10b67b0be8f' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/stream.php',
|
||||||
|
'764b09f6df081cbb2807b97c6ace3866' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/strings.php',
|
||||||
|
'ef241678769fee4a44aaa288f3b78aa1' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/swoole.php',
|
||||||
|
'0efc8f6778cba932b9e2a89e28de2452' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/uodbc.php',
|
||||||
|
'd383d32907b98af53ee9208c62204fd0' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/uopz.php',
|
||||||
|
'2fd2e4060f7fe772660f002ce38f0b71' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/url.php',
|
||||||
|
'782249e03deebeaf57b9991ff5493aa0' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/var.php',
|
||||||
|
'344440cd1cd7200fdb4f12af0d3c587f' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/xdiff.php',
|
||||||
|
'3599f369219c658a5fb6c4fe66832f62' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/xml.php',
|
||||||
|
'7fcd313da9fae337051b091b3492c21b' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/xmlrpc.php',
|
||||||
|
'd668c74cfa92d893b582356733d9a80e' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/yaml.php',
|
||||||
|
'4af1dca6db8c527c6eed27bff85ff0e5' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/yaz.php',
|
||||||
|
'fe43ca06499ac37bc2dedd823af71eb5' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/zip.php',
|
||||||
|
'356736db98a6834f0a886b8d509b0ecd' => __DIR__ . '/..' . '/thecodingmachine/safe/generated/zlib.php',
|
||||||
'e69f7f6ee287b969198c3c9d6777bd38' => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
'e69f7f6ee287b969198c3c9d6777bd38' => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer/bootstrap.php',
|
||||||
'023d27dca8066ef29e6739335ea73bad' => __DIR__ . '/..' . '/symfony/polyfill-php70/bootstrap.php',
|
|
||||||
'25072dd6e2470089de65ae7bf11d3109' => __DIR__ . '/..' . '/symfony/polyfill-php72/bootstrap.php',
|
'25072dd6e2470089de65ae7bf11d3109' => __DIR__ . '/..' . '/symfony/polyfill-php72/bootstrap.php',
|
||||||
'c964ee0ededf28c96ebd9db5099ef910' => __DIR__ . '/..' . '/guzzlehttp/promises/src/functions_include.php',
|
'8825ede83f2f289127722d4e842cf7e8' => __DIR__ . '/..' . '/symfony/polyfill-intl-grapheme/bootstrap.php',
|
||||||
'a0edc8309cc5e1d60e3047b5df6b7052' => __DIR__ . '/..' . '/guzzlehttp/psr7/src/functions_include.php',
|
|
||||||
'f598d06aa772fa33d905e87be6398fb1' => __DIR__ . '/..' . '/symfony/polyfill-intl-idn/bootstrap.php',
|
'f598d06aa772fa33d905e87be6398fb1' => __DIR__ . '/..' . '/symfony/polyfill-intl-idn/bootstrap.php',
|
||||||
'37a3dc5111fe8f707ab4c132ef1dbc62' => __DIR__ . '/..' . '/guzzlehttp/guzzle/src/functions_include.php',
|
'0d59ee240a4cd96ddbb4ff164fccea4d' => __DIR__ . '/..' . '/symfony/polyfill-php73/bootstrap.php',
|
||||||
);
|
);
|
||||||
|
|
||||||
public static $prefixLengthsPsr4 = array (
|
public static $prefixLengthsPsr4 = array (
|
||||||
'S' =>
|
'S' =>
|
||||||
array (
|
array (
|
||||||
|
'Symfony\\Polyfill\\Php73\\' => 23,
|
||||||
'Symfony\\Polyfill\\Php72\\' => 23,
|
'Symfony\\Polyfill\\Php72\\' => 23,
|
||||||
'Symfony\\Polyfill\\Php70\\' => 23,
|
|
||||||
'Symfony\\Polyfill\\Intl\\Normalizer\\' => 33,
|
'Symfony\\Polyfill\\Intl\\Normalizer\\' => 33,
|
||||||
'Symfony\\Polyfill\\Intl\\Idn\\' => 26,
|
'Symfony\\Polyfill\\Intl\\Idn\\' => 26,
|
||||||
|
'Symfony\\Polyfill\\Intl\\Grapheme\\' => 31,
|
||||||
|
'Safe\\' => 5,
|
||||||
),
|
),
|
||||||
'P' =>
|
'P' =>
|
||||||
array (
|
array (
|
||||||
'Psr\\Http\\Message\\' => 17,
|
'Psr\\SimpleCache\\' => 16,
|
||||||
|
'Psr\\Log\\' => 8,
|
||||||
|
'Psr\\EventDispatcher\\' => 20,
|
||||||
|
'Psr\\Container\\' => 14,
|
||||||
),
|
),
|
||||||
'N' =>
|
'N' =>
|
||||||
array (
|
array (
|
||||||
'Nuwave\\Lighthouse\\' => 18,
|
'Nuwave\\Lighthouse\\' => 18,
|
||||||
),
|
),
|
||||||
|
'L' =>
|
||||||
|
array (
|
||||||
|
'Laragraph\\Utils\\' => 16,
|
||||||
|
),
|
||||||
|
'H' =>
|
||||||
|
array (
|
||||||
|
'HaydenPierce\\ClassFinder\\UnitTest\\' => 34,
|
||||||
|
'HaydenPierce\\ClassFinder\\' => 25,
|
||||||
|
),
|
||||||
'G' =>
|
'G' =>
|
||||||
array (
|
array (
|
||||||
'GuzzleHttp\\Psr7\\' => 16,
|
|
||||||
'GuzzleHttp\\Promise\\' => 19,
|
|
||||||
'GuzzleHttp\\' => 11,
|
|
||||||
'GraphQL\\' => 8,
|
'GraphQL\\' => 8,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
public static $prefixDirsPsr4 = array (
|
public static $prefixDirsPsr4 = array (
|
||||||
|
'Symfony\\Polyfill\\Php73\\' =>
|
||||||
|
array (
|
||||||
|
0 => __DIR__ . '/..' . '/symfony/polyfill-php73',
|
||||||
|
),
|
||||||
'Symfony\\Polyfill\\Php72\\' =>
|
'Symfony\\Polyfill\\Php72\\' =>
|
||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/symfony/polyfill-php72',
|
0 => __DIR__ . '/..' . '/symfony/polyfill-php72',
|
||||||
),
|
),
|
||||||
'Symfony\\Polyfill\\Php70\\' =>
|
|
||||||
array (
|
|
||||||
0 => __DIR__ . '/..' . '/symfony/polyfill-php70',
|
|
||||||
),
|
|
||||||
'Symfony\\Polyfill\\Intl\\Normalizer\\' =>
|
'Symfony\\Polyfill\\Intl\\Normalizer\\' =>
|
||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer',
|
0 => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer',
|
||||||
@@ -59,25 +156,47 @@ class ComposerStaticInit02b791b67b928853969b061eb6816088
|
|||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/symfony/polyfill-intl-idn',
|
0 => __DIR__ . '/..' . '/symfony/polyfill-intl-idn',
|
||||||
),
|
),
|
||||||
'Psr\\Http\\Message\\' =>
|
'Symfony\\Polyfill\\Intl\\Grapheme\\' =>
|
||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/psr/http-message/src',
|
0 => __DIR__ . '/..' . '/symfony/polyfill-intl-grapheme',
|
||||||
|
),
|
||||||
|
'Safe\\' =>
|
||||||
|
array (
|
||||||
|
0 => __DIR__ . '/..' . '/thecodingmachine/safe/lib',
|
||||||
|
1 => __DIR__ . '/..' . '/thecodingmachine/safe/deprecated',
|
||||||
|
2 => __DIR__ . '/..' . '/thecodingmachine/safe/generated',
|
||||||
|
),
|
||||||
|
'Psr\\SimpleCache\\' =>
|
||||||
|
array (
|
||||||
|
0 => __DIR__ . '/..' . '/psr/simple-cache/src',
|
||||||
|
),
|
||||||
|
'Psr\\Log\\' =>
|
||||||
|
array (
|
||||||
|
0 => __DIR__ . '/..' . '/psr/log/Psr/Log',
|
||||||
|
),
|
||||||
|
'Psr\\EventDispatcher\\' =>
|
||||||
|
array (
|
||||||
|
0 => __DIR__ . '/..' . '/psr/event-dispatcher/src',
|
||||||
|
),
|
||||||
|
'Psr\\Container\\' =>
|
||||||
|
array (
|
||||||
|
0 => __DIR__ . '/..' . '/psr/container/src',
|
||||||
),
|
),
|
||||||
'Nuwave\\Lighthouse\\' =>
|
'Nuwave\\Lighthouse\\' =>
|
||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/nuwave/lighthouse/src',
|
0 => __DIR__ . '/..' . '/nuwave/lighthouse/src',
|
||||||
),
|
),
|
||||||
'GuzzleHttp\\Psr7\\' =>
|
'Laragraph\\Utils\\' =>
|
||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/guzzlehttp/psr7/src',
|
0 => __DIR__ . '/..' . '/laragraph/utils/src',
|
||||||
),
|
),
|
||||||
'GuzzleHttp\\Promise\\' =>
|
'HaydenPierce\\ClassFinder\\UnitTest\\' =>
|
||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/guzzlehttp/promises/src',
|
0 => __DIR__ . '/..' . '/haydenpierce/class-finder/test/unit',
|
||||||
),
|
),
|
||||||
'GuzzleHttp\\' =>
|
'HaydenPierce\\ClassFinder\\' =>
|
||||||
array (
|
array (
|
||||||
0 => __DIR__ . '/..' . '/guzzlehttp/guzzle/src',
|
0 => __DIR__ . '/..' . '/haydenpierce/class-finder/src',
|
||||||
),
|
),
|
||||||
'GraphQL\\' =>
|
'GraphQL\\' =>
|
||||||
array (
|
array (
|
||||||
@@ -86,14 +205,9 @@ class ComposerStaticInit02b791b67b928853969b061eb6816088
|
|||||||
);
|
);
|
||||||
|
|
||||||
public static $classMap = array (
|
public static $classMap = array (
|
||||||
'ArithmeticError' => __DIR__ . '/..' . '/symfony/polyfill-php70/Resources/stubs/ArithmeticError.php',
|
'Composer\\InstalledVersions' => __DIR__ . '/..' . '/composer/InstalledVersions.php',
|
||||||
'AssertionError' => __DIR__ . '/..' . '/symfony/polyfill-php70/Resources/stubs/AssertionError.php',
|
'JsonException' => __DIR__ . '/..' . '/symfony/polyfill-php73/Resources/stubs/JsonException.php',
|
||||||
'DivisionByZeroError' => __DIR__ . '/..' . '/symfony/polyfill-php70/Resources/stubs/DivisionByZeroError.php',
|
|
||||||
'Error' => __DIR__ . '/..' . '/symfony/polyfill-php70/Resources/stubs/Error.php',
|
|
||||||
'Normalizer' => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer/Resources/stubs/Normalizer.php',
|
'Normalizer' => __DIR__ . '/..' . '/symfony/polyfill-intl-normalizer/Resources/stubs/Normalizer.php',
|
||||||
'ParseError' => __DIR__ . '/..' . '/symfony/polyfill-php70/Resources/stubs/ParseError.php',
|
|
||||||
'SessionUpdateTimestampHandlerInterface' => __DIR__ . '/..' . '/symfony/polyfill-php70/Resources/stubs/SessionUpdateTimestampHandlerInterface.php',
|
|
||||||
'TypeError' => __DIR__ . '/..' . '/symfony/polyfill-php70/Resources/stubs/TypeError.php',
|
|
||||||
);
|
);
|
||||||
|
|
||||||
public static function getInitializer(ClassLoader $loader)
|
public static function getInitializer(ClassLoader $loader)
|
||||||
|
|||||||
Vendored
+4706
-729
File diff suppressed because it is too large
Load Diff
Vendored
+26
@@ -0,0 +1,26 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
// platform_check.php @generated by Composer
|
||||||
|
|
||||||
|
$issues = array();
|
||||||
|
|
||||||
|
if (!(PHP_VERSION_ID >= 70200)) {
|
||||||
|
$issues[] = 'Your Composer dependencies require a PHP version ">= 7.2.0". You are running ' . PHP_VERSION . '.';
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($issues) {
|
||||||
|
if (!headers_sent()) {
|
||||||
|
header('HTTP/1.1 500 Internal Server Error');
|
||||||
|
}
|
||||||
|
if (!ini_get('display_errors')) {
|
||||||
|
if (PHP_SAPI === 'cli' || PHP_SAPI === 'phpdbg') {
|
||||||
|
fwrite(STDERR, 'Composer detected issues in your platform:' . PHP_EOL.PHP_EOL . implode(PHP_EOL, $issues) . PHP_EOL.PHP_EOL);
|
||||||
|
} elseif (!headers_sent()) {
|
||||||
|
echo 'Composer detected issues in your platform:' . PHP_EOL.PHP_EOL . str_replace('You are running '.PHP_VERSION.'.', '', implode(PHP_EOL, $issues)) . PHP_EOL.PHP_EOL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
trigger_error(
|
||||||
|
'Composer detected issues in your platform: ' . implode(' ', $issues),
|
||||||
|
E_USER_ERROR
|
||||||
|
);
|
||||||
|
}
|
||||||
-1338
File diff suppressed because it is too large
Load Diff
Vendored
-18
@@ -1,18 +0,0 @@
|
|||||||
FROM composer:latest as setup
|
|
||||||
|
|
||||||
RUN mkdir /guzzle
|
|
||||||
|
|
||||||
WORKDIR /guzzle
|
|
||||||
|
|
||||||
RUN set -xe \
|
|
||||||
&& composer init --name=guzzlehttp/test --description="Simple project for testing Guzzle scripts" --author="Márk Sági-Kazár <mark.sagikazar@gmail.com>" --no-interaction \
|
|
||||||
&& composer require guzzlehttp/guzzle
|
|
||||||
|
|
||||||
|
|
||||||
FROM php:7.3
|
|
||||||
|
|
||||||
RUN mkdir /guzzle
|
|
||||||
|
|
||||||
WORKDIR /guzzle
|
|
||||||
|
|
||||||
COPY --from=setup /guzzle /guzzle
|
|
||||||
Vendored
-19
@@ -1,19 +0,0 @@
|
|||||||
Copyright (c) 2011-2018 Michael Dowling, https://github.com/mtdowling <mtdowling@gmail.com>
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in
|
|
||||||
all copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
THE SOFTWARE.
|
|
||||||
Vendored
-90
@@ -1,90 +0,0 @@
|
|||||||
Guzzle, PHP HTTP client
|
|
||||||
=======================
|
|
||||||
|
|
||||||
[](https://github.com/guzzle/guzzle/releases)
|
|
||||||
[](https://travis-ci.org/guzzle/guzzle)
|
|
||||||
[](https://packagist.org/packages/guzzlehttp/guzzle)
|
|
||||||
|
|
||||||
Guzzle is a PHP HTTP client that makes it easy to send HTTP requests and
|
|
||||||
trivial to integrate with web services.
|
|
||||||
|
|
||||||
- Simple interface for building query strings, POST requests, streaming large
|
|
||||||
uploads, streaming large downloads, using HTTP cookies, uploading JSON data,
|
|
||||||
etc...
|
|
||||||
- Can send both synchronous and asynchronous requests using the same interface.
|
|
||||||
- Uses PSR-7 interfaces for requests, responses, and streams. This allows you
|
|
||||||
to utilize other PSR-7 compatible libraries with Guzzle.
|
|
||||||
- Abstracts away the underlying HTTP transport, allowing you to write
|
|
||||||
environment and transport agnostic code; i.e., no hard dependency on cURL,
|
|
||||||
PHP streams, sockets, or non-blocking event loops.
|
|
||||||
- Middleware system allows you to augment and compose client behavior.
|
|
||||||
|
|
||||||
```php
|
|
||||||
$client = new \GuzzleHttp\Client();
|
|
||||||
$response = $client->request('GET', 'https://api.github.com/repos/guzzle/guzzle');
|
|
||||||
|
|
||||||
echo $response->getStatusCode(); # 200
|
|
||||||
echo $response->getHeaderLine('content-type'); # 'application/json; charset=utf8'
|
|
||||||
echo $response->getBody(); # '{"id": 1420053, "name": "guzzle", ...}'
|
|
||||||
|
|
||||||
# Send an asynchronous request.
|
|
||||||
$request = new \GuzzleHttp\Psr7\Request('GET', 'http://httpbin.org');
|
|
||||||
$promise = $client->sendAsync($request)->then(function ($response) {
|
|
||||||
echo 'I completed! ' . $response->getBody();
|
|
||||||
});
|
|
||||||
|
|
||||||
$promise->wait();
|
|
||||||
```
|
|
||||||
|
|
||||||
## Help and docs
|
|
||||||
|
|
||||||
- [Documentation](http://guzzlephp.org/)
|
|
||||||
- [Stack Overflow](http://stackoverflow.com/questions/tagged/guzzle)
|
|
||||||
- [Gitter](https://gitter.im/guzzle/guzzle)
|
|
||||||
|
|
||||||
|
|
||||||
## Installing Guzzle
|
|
||||||
|
|
||||||
The recommended way to install Guzzle is through
|
|
||||||
[Composer](http://getcomposer.org).
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Install Composer
|
|
||||||
curl -sS https://getcomposer.org/installer | php
|
|
||||||
```
|
|
||||||
|
|
||||||
Next, run the Composer command to install the latest stable version of Guzzle:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
composer require guzzlehttp/guzzle
|
|
||||||
```
|
|
||||||
|
|
||||||
After installing, you need to require Composer's autoloader:
|
|
||||||
|
|
||||||
```php
|
|
||||||
require 'vendor/autoload.php';
|
|
||||||
```
|
|
||||||
|
|
||||||
You can then later update Guzzle using composer:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
composer update
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
## Version Guidance
|
|
||||||
|
|
||||||
| Version | Status | Packagist | Namespace | Repo | Docs | PSR-7 | PHP Version |
|
|
||||||
|---------|------------|---------------------|--------------|---------------------|---------------------|-------|-------------|
|
|
||||||
| 3.x | EOL | `guzzle/guzzle` | `Guzzle` | [v3][guzzle-3-repo] | [v3][guzzle-3-docs] | No | >= 5.3.3 |
|
|
||||||
| 4.x | EOL | `guzzlehttp/guzzle` | `GuzzleHttp` | [v4][guzzle-4-repo] | N/A | No | >= 5.4 |
|
|
||||||
| 5.x | EOL | `guzzlehttp/guzzle` | `GuzzleHttp` | [v5][guzzle-5-repo] | [v5][guzzle-5-docs] | No | >= 5.4 |
|
|
||||||
| 6.x | Latest | `guzzlehttp/guzzle` | `GuzzleHttp` | [v6][guzzle-6-repo] | [v6][guzzle-6-docs] | Yes | >= 5.5 |
|
|
||||||
|
|
||||||
[guzzle-3-repo]: https://github.com/guzzle/guzzle3
|
|
||||||
[guzzle-4-repo]: https://github.com/guzzle/guzzle/tree/4.x
|
|
||||||
[guzzle-5-repo]: https://github.com/guzzle/guzzle/tree/5.3
|
|
||||||
[guzzle-6-repo]: https://github.com/guzzle/guzzle
|
|
||||||
[guzzle-3-docs]: http://guzzle3.readthedocs.org
|
|
||||||
[guzzle-5-docs]: http://guzzle.readthedocs.org/en/5.3/
|
|
||||||
[guzzle-6-docs]: http://guzzle.readthedocs.org/en/latest/
|
|
||||||
-1203
File diff suppressed because it is too large
Load Diff
-59
@@ -1,59 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "guzzlehttp/guzzle",
|
|
||||||
"type": "library",
|
|
||||||
"description": "Guzzle is a PHP HTTP client library",
|
|
||||||
"keywords": [
|
|
||||||
"framework",
|
|
||||||
"http",
|
|
||||||
"rest",
|
|
||||||
"web service",
|
|
||||||
"curl",
|
|
||||||
"client",
|
|
||||||
"HTTP client"
|
|
||||||
],
|
|
||||||
"homepage": "http://guzzlephp.org/",
|
|
||||||
"license": "MIT",
|
|
||||||
"authors": [
|
|
||||||
{
|
|
||||||
"name": "Michael Dowling",
|
|
||||||
"email": "mtdowling@gmail.com",
|
|
||||||
"homepage": "https://github.com/mtdowling"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"require": {
|
|
||||||
"php": ">=5.5",
|
|
||||||
"ext-json": "*",
|
|
||||||
"symfony/polyfill-intl-idn": "^1.17.0",
|
|
||||||
"guzzlehttp/promises": "^1.0",
|
|
||||||
"guzzlehttp/psr7": "^1.6.1"
|
|
||||||
},
|
|
||||||
"require-dev": {
|
|
||||||
"ext-curl": "*",
|
|
||||||
"phpunit/phpunit": "^4.8.35 || ^5.7 || ^6.4 || ^7.0",
|
|
||||||
"psr/log": "^1.1"
|
|
||||||
},
|
|
||||||
"suggest": {
|
|
||||||
"psr/log": "Required for using the Log middleware"
|
|
||||||
},
|
|
||||||
"config": {
|
|
||||||
"sort-packages": true
|
|
||||||
},
|
|
||||||
"extra": {
|
|
||||||
"branch-alias": {
|
|
||||||
"dev-master": "6.5-dev"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"autoload": {
|
|
||||||
"psr-4": {
|
|
||||||
"GuzzleHttp\\": "src/"
|
|
||||||
},
|
|
||||||
"files": [
|
|
||||||
"src/functions_include.php"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"autoload-dev": {
|
|
||||||
"psr-4": {
|
|
||||||
"GuzzleHttp\\Tests\\": "tests/"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-501
@@ -1,501 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Cookie\CookieJar;
|
|
||||||
use GuzzleHttp\Exception\GuzzleException;
|
|
||||||
use GuzzleHttp\Promise;
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Http\Message\UriInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @method ResponseInterface get(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method ResponseInterface head(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method ResponseInterface put(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method ResponseInterface post(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method ResponseInterface patch(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method ResponseInterface delete(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method Promise\PromiseInterface getAsync(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method Promise\PromiseInterface headAsync(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method Promise\PromiseInterface putAsync(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method Promise\PromiseInterface postAsync(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method Promise\PromiseInterface patchAsync(string|UriInterface $uri, array $options = [])
|
|
||||||
* @method Promise\PromiseInterface deleteAsync(string|UriInterface $uri, array $options = [])
|
|
||||||
*/
|
|
||||||
class Client implements ClientInterface
|
|
||||||
{
|
|
||||||
/** @var array Default request options */
|
|
||||||
private $config;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Clients accept an array of constructor parameters.
|
|
||||||
*
|
|
||||||
* Here's an example of creating a client using a base_uri and an array of
|
|
||||||
* default request options to apply to each request:
|
|
||||||
*
|
|
||||||
* $client = new Client([
|
|
||||||
* 'base_uri' => 'http://www.foo.com/1.0/',
|
|
||||||
* 'timeout' => 0,
|
|
||||||
* 'allow_redirects' => false,
|
|
||||||
* 'proxy' => '192.168.16.1:10'
|
|
||||||
* ]);
|
|
||||||
*
|
|
||||||
* Client configuration settings include the following options:
|
|
||||||
*
|
|
||||||
* - handler: (callable) Function that transfers HTTP requests over the
|
|
||||||
* wire. The function is called with a Psr7\Http\Message\RequestInterface
|
|
||||||
* and array of transfer options, and must return a
|
|
||||||
* GuzzleHttp\Promise\PromiseInterface that is fulfilled with a
|
|
||||||
* Psr7\Http\Message\ResponseInterface on success.
|
|
||||||
* If no handler is provided, a default handler will be created
|
|
||||||
* that enables all of the request options below by attaching all of the
|
|
||||||
* default middleware to the handler.
|
|
||||||
* - base_uri: (string|UriInterface) Base URI of the client that is merged
|
|
||||||
* into relative URIs. Can be a string or instance of UriInterface.
|
|
||||||
* - **: any request option
|
|
||||||
*
|
|
||||||
* @param array $config Client configuration settings.
|
|
||||||
*
|
|
||||||
* @see \GuzzleHttp\RequestOptions for a list of available request options.
|
|
||||||
*/
|
|
||||||
public function __construct(array $config = [])
|
|
||||||
{
|
|
||||||
if (!isset($config['handler'])) {
|
|
||||||
$config['handler'] = HandlerStack::create();
|
|
||||||
} elseif (!is_callable($config['handler'])) {
|
|
||||||
throw new \InvalidArgumentException('handler must be a callable');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert the base_uri to a UriInterface
|
|
||||||
if (isset($config['base_uri'])) {
|
|
||||||
$config['base_uri'] = Psr7\uri_for($config['base_uri']);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->configureDefaults($config);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $method
|
|
||||||
* @param array $args
|
|
||||||
*
|
|
||||||
* @return Promise\PromiseInterface
|
|
||||||
*/
|
|
||||||
public function __call($method, $args)
|
|
||||||
{
|
|
||||||
if (count($args) < 1) {
|
|
||||||
throw new \InvalidArgumentException('Magic request methods require a URI and optional options array');
|
|
||||||
}
|
|
||||||
|
|
||||||
$uri = $args[0];
|
|
||||||
$opts = isset($args[1]) ? $args[1] : [];
|
|
||||||
|
|
||||||
return substr($method, -5) === 'Async'
|
|
||||||
? $this->requestAsync(substr($method, 0, -5), $uri, $opts)
|
|
||||||
: $this->request($method, $uri, $opts);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Asynchronously send an HTTP request.
|
|
||||||
*
|
|
||||||
* @param array $options Request options to apply to the given
|
|
||||||
* request and to the transfer. See \GuzzleHttp\RequestOptions.
|
|
||||||
*
|
|
||||||
* @return Promise\PromiseInterface
|
|
||||||
*/
|
|
||||||
public function sendAsync(RequestInterface $request, array $options = [])
|
|
||||||
{
|
|
||||||
// Merge the base URI into the request URI if needed.
|
|
||||||
$options = $this->prepareDefaults($options);
|
|
||||||
|
|
||||||
return $this->transfer(
|
|
||||||
$request->withUri($this->buildUri($request->getUri(), $options), $request->hasHeader('Host')),
|
|
||||||
$options
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send an HTTP request.
|
|
||||||
*
|
|
||||||
* @param array $options Request options to apply to the given
|
|
||||||
* request and to the transfer. See \GuzzleHttp\RequestOptions.
|
|
||||||
*
|
|
||||||
* @return ResponseInterface
|
|
||||||
* @throws GuzzleException
|
|
||||||
*/
|
|
||||||
public function send(RequestInterface $request, array $options = [])
|
|
||||||
{
|
|
||||||
$options[RequestOptions::SYNCHRONOUS] = true;
|
|
||||||
return $this->sendAsync($request, $options)->wait();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create and send an asynchronous HTTP request.
|
|
||||||
*
|
|
||||||
* Use an absolute path to override the base path of the client, or a
|
|
||||||
* relative path to append to the base path of the client. The URL can
|
|
||||||
* contain the query string as well. Use an array to provide a URL
|
|
||||||
* template and additional variables to use in the URL template expansion.
|
|
||||||
*
|
|
||||||
* @param string $method HTTP method
|
|
||||||
* @param string|UriInterface $uri URI object or string.
|
|
||||||
* @param array $options Request options to apply. See \GuzzleHttp\RequestOptions.
|
|
||||||
*
|
|
||||||
* @return Promise\PromiseInterface
|
|
||||||
*/
|
|
||||||
public function requestAsync($method, $uri = '', array $options = [])
|
|
||||||
{
|
|
||||||
$options = $this->prepareDefaults($options);
|
|
||||||
// Remove request modifying parameter because it can be done up-front.
|
|
||||||
$headers = isset($options['headers']) ? $options['headers'] : [];
|
|
||||||
$body = isset($options['body']) ? $options['body'] : null;
|
|
||||||
$version = isset($options['version']) ? $options['version'] : '1.1';
|
|
||||||
// Merge the URI into the base URI.
|
|
||||||
$uri = $this->buildUri($uri, $options);
|
|
||||||
if (is_array($body)) {
|
|
||||||
$this->invalidBody();
|
|
||||||
}
|
|
||||||
$request = new Psr7\Request($method, $uri, $headers, $body, $version);
|
|
||||||
// Remove the option so that they are not doubly-applied.
|
|
||||||
unset($options['headers'], $options['body'], $options['version']);
|
|
||||||
|
|
||||||
return $this->transfer($request, $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create and send an HTTP request.
|
|
||||||
*
|
|
||||||
* Use an absolute path to override the base path of the client, or a
|
|
||||||
* relative path to append to the base path of the client. The URL can
|
|
||||||
* contain the query string as well.
|
|
||||||
*
|
|
||||||
* @param string $method HTTP method.
|
|
||||||
* @param string|UriInterface $uri URI object or string.
|
|
||||||
* @param array $options Request options to apply. See \GuzzleHttp\RequestOptions.
|
|
||||||
*
|
|
||||||
* @return ResponseInterface
|
|
||||||
* @throws GuzzleException
|
|
||||||
*/
|
|
||||||
public function request($method, $uri = '', array $options = [])
|
|
||||||
{
|
|
||||||
$options[RequestOptions::SYNCHRONOUS] = true;
|
|
||||||
return $this->requestAsync($method, $uri, $options)->wait();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a client configuration option.
|
|
||||||
*
|
|
||||||
* These options include default request options of the client, a "handler"
|
|
||||||
* (if utilized by the concrete client), and a "base_uri" if utilized by
|
|
||||||
* the concrete client.
|
|
||||||
*
|
|
||||||
* @param string|null $option The config option to retrieve.
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function getConfig($option = null)
|
|
||||||
{
|
|
||||||
return $option === null
|
|
||||||
? $this->config
|
|
||||||
: (isset($this->config[$option]) ? $this->config[$option] : null);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string|null $uri
|
|
||||||
*
|
|
||||||
* @return UriInterface
|
|
||||||
*/
|
|
||||||
private function buildUri($uri, array $config)
|
|
||||||
{
|
|
||||||
// for BC we accept null which would otherwise fail in uri_for
|
|
||||||
$uri = Psr7\uri_for($uri === null ? '' : $uri);
|
|
||||||
|
|
||||||
if (isset($config['base_uri'])) {
|
|
||||||
$uri = Psr7\UriResolver::resolve(Psr7\uri_for($config['base_uri']), $uri);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($config['idn_conversion']) && ($config['idn_conversion'] !== false)) {
|
|
||||||
$idnOptions = ($config['idn_conversion'] === true) ? IDNA_DEFAULT : $config['idn_conversion'];
|
|
||||||
$uri = Utils::idnUriConvert($uri, $idnOptions);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $uri->getScheme() === '' && $uri->getHost() !== '' ? $uri->withScheme('http') : $uri;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Configures the default options for a client.
|
|
||||||
*
|
|
||||||
* @param array $config
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
private function configureDefaults(array $config)
|
|
||||||
{
|
|
||||||
$defaults = [
|
|
||||||
'allow_redirects' => RedirectMiddleware::$defaultSettings,
|
|
||||||
'http_errors' => true,
|
|
||||||
'decode_content' => true,
|
|
||||||
'verify' => true,
|
|
||||||
'cookies' => false,
|
|
||||||
'idn_conversion' => true,
|
|
||||||
];
|
|
||||||
|
|
||||||
// Use the standard Linux HTTP_PROXY and HTTPS_PROXY if set.
|
|
||||||
|
|
||||||
// We can only trust the HTTP_PROXY environment variable in a CLI
|
|
||||||
// process due to the fact that PHP has no reliable mechanism to
|
|
||||||
// get environment variables that start with "HTTP_".
|
|
||||||
if (php_sapi_name() === 'cli' && getenv('HTTP_PROXY')) {
|
|
||||||
$defaults['proxy']['http'] = getenv('HTTP_PROXY');
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($proxy = getenv('HTTPS_PROXY')) {
|
|
||||||
$defaults['proxy']['https'] = $proxy;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($noProxy = getenv('NO_PROXY')) {
|
|
||||||
$cleanedNoProxy = str_replace(' ', '', $noProxy);
|
|
||||||
$defaults['proxy']['no'] = explode(',', $cleanedNoProxy);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->config = $config + $defaults;
|
|
||||||
|
|
||||||
if (!empty($config['cookies']) && $config['cookies'] === true) {
|
|
||||||
$this->config['cookies'] = new CookieJar();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add the default user-agent header.
|
|
||||||
if (!isset($this->config['headers'])) {
|
|
||||||
$this->config['headers'] = ['User-Agent' => default_user_agent()];
|
|
||||||
} else {
|
|
||||||
// Add the User-Agent header if one was not already set.
|
|
||||||
foreach (array_keys($this->config['headers']) as $name) {
|
|
||||||
if (strtolower($name) === 'user-agent') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$this->config['headers']['User-Agent'] = default_user_agent();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Merges default options into the array.
|
|
||||||
*
|
|
||||||
* @param array $options Options to modify by reference
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
private function prepareDefaults(array $options)
|
|
||||||
{
|
|
||||||
$defaults = $this->config;
|
|
||||||
|
|
||||||
if (!empty($defaults['headers'])) {
|
|
||||||
// Default headers are only added if they are not present.
|
|
||||||
$defaults['_conditional'] = $defaults['headers'];
|
|
||||||
unset($defaults['headers']);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Special handling for headers is required as they are added as
|
|
||||||
// conditional headers and as headers passed to a request ctor.
|
|
||||||
if (array_key_exists('headers', $options)) {
|
|
||||||
// Allows default headers to be unset.
|
|
||||||
if ($options['headers'] === null) {
|
|
||||||
$defaults['_conditional'] = [];
|
|
||||||
unset($options['headers']);
|
|
||||||
} elseif (!is_array($options['headers'])) {
|
|
||||||
throw new \InvalidArgumentException('headers must be an array');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Shallow merge defaults underneath options.
|
|
||||||
$result = $options + $defaults;
|
|
||||||
|
|
||||||
// Remove null values.
|
|
||||||
foreach ($result as $k => $v) {
|
|
||||||
if ($v === null) {
|
|
||||||
unset($result[$k]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Transfers the given request and applies request options.
|
|
||||||
*
|
|
||||||
* The URI of the request is not modified and the request options are used
|
|
||||||
* as-is without merging in default options.
|
|
||||||
*
|
|
||||||
* @param array $options See \GuzzleHttp\RequestOptions.
|
|
||||||
*
|
|
||||||
* @return Promise\PromiseInterface
|
|
||||||
*/
|
|
||||||
private function transfer(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
// save_to -> sink
|
|
||||||
if (isset($options['save_to'])) {
|
|
||||||
$options['sink'] = $options['save_to'];
|
|
||||||
unset($options['save_to']);
|
|
||||||
}
|
|
||||||
|
|
||||||
// exceptions -> http_errors
|
|
||||||
if (isset($options['exceptions'])) {
|
|
||||||
$options['http_errors'] = $options['exceptions'];
|
|
||||||
unset($options['exceptions']);
|
|
||||||
}
|
|
||||||
|
|
||||||
$request = $this->applyOptions($request, $options);
|
|
||||||
/** @var HandlerStack $handler */
|
|
||||||
$handler = $options['handler'];
|
|
||||||
|
|
||||||
try {
|
|
||||||
return Promise\promise_for($handler($request, $options));
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
return Promise\rejection_for($e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Applies the array of request options to a request.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param array $options
|
|
||||||
*
|
|
||||||
* @return RequestInterface
|
|
||||||
*/
|
|
||||||
private function applyOptions(RequestInterface $request, array &$options)
|
|
||||||
{
|
|
||||||
$modify = [
|
|
||||||
'set_headers' => [],
|
|
||||||
];
|
|
||||||
|
|
||||||
if (isset($options['headers'])) {
|
|
||||||
$modify['set_headers'] = $options['headers'];
|
|
||||||
unset($options['headers']);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['form_params'])) {
|
|
||||||
if (isset($options['multipart'])) {
|
|
||||||
throw new \InvalidArgumentException('You cannot use '
|
|
||||||
. 'form_params and multipart at the same time. Use the '
|
|
||||||
. 'form_params option if you want to send application/'
|
|
||||||
. 'x-www-form-urlencoded requests, and the multipart '
|
|
||||||
. 'option to send multipart/form-data requests.');
|
|
||||||
}
|
|
||||||
$options['body'] = http_build_query($options['form_params'], '', '&');
|
|
||||||
unset($options['form_params']);
|
|
||||||
// Ensure that we don't have the header in different case and set the new value.
|
|
||||||
$options['_conditional'] = Psr7\_caseless_remove(['Content-Type'], $options['_conditional']);
|
|
||||||
$options['_conditional']['Content-Type'] = 'application/x-www-form-urlencoded';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['multipart'])) {
|
|
||||||
$options['body'] = new Psr7\MultipartStream($options['multipart']);
|
|
||||||
unset($options['multipart']);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['json'])) {
|
|
||||||
$options['body'] = \GuzzleHttp\json_encode($options['json']);
|
|
||||||
unset($options['json']);
|
|
||||||
// Ensure that we don't have the header in different case and set the new value.
|
|
||||||
$options['_conditional'] = Psr7\_caseless_remove(['Content-Type'], $options['_conditional']);
|
|
||||||
$options['_conditional']['Content-Type'] = 'application/json';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($options['decode_content'])
|
|
||||||
&& $options['decode_content'] !== true
|
|
||||||
) {
|
|
||||||
// Ensure that we don't have the header in different case and set the new value.
|
|
||||||
$options['_conditional'] = Psr7\_caseless_remove(['Accept-Encoding'], $options['_conditional']);
|
|
||||||
$modify['set_headers']['Accept-Encoding'] = $options['decode_content'];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['body'])) {
|
|
||||||
if (is_array($options['body'])) {
|
|
||||||
$this->invalidBody();
|
|
||||||
}
|
|
||||||
$modify['body'] = Psr7\stream_for($options['body']);
|
|
||||||
unset($options['body']);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($options['auth']) && is_array($options['auth'])) {
|
|
||||||
$value = $options['auth'];
|
|
||||||
$type = isset($value[2]) ? strtolower($value[2]) : 'basic';
|
|
||||||
switch ($type) {
|
|
||||||
case 'basic':
|
|
||||||
// Ensure that we don't have the header in different case and set the new value.
|
|
||||||
$modify['set_headers'] = Psr7\_caseless_remove(['Authorization'], $modify['set_headers']);
|
|
||||||
$modify['set_headers']['Authorization'] = 'Basic '
|
|
||||||
. base64_encode("$value[0]:$value[1]");
|
|
||||||
break;
|
|
||||||
case 'digest':
|
|
||||||
// @todo: Do not rely on curl
|
|
||||||
$options['curl'][CURLOPT_HTTPAUTH] = CURLAUTH_DIGEST;
|
|
||||||
$options['curl'][CURLOPT_USERPWD] = "$value[0]:$value[1]";
|
|
||||||
break;
|
|
||||||
case 'ntlm':
|
|
||||||
$options['curl'][CURLOPT_HTTPAUTH] = CURLAUTH_NTLM;
|
|
||||||
$options['curl'][CURLOPT_USERPWD] = "$value[0]:$value[1]";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['query'])) {
|
|
||||||
$value = $options['query'];
|
|
||||||
if (is_array($value)) {
|
|
||||||
$value = http_build_query($value, null, '&', PHP_QUERY_RFC3986);
|
|
||||||
}
|
|
||||||
if (!is_string($value)) {
|
|
||||||
throw new \InvalidArgumentException('query must be a string or array');
|
|
||||||
}
|
|
||||||
$modify['query'] = $value;
|
|
||||||
unset($options['query']);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure that sink is not an invalid value.
|
|
||||||
if (isset($options['sink'])) {
|
|
||||||
// TODO: Add more sink validation?
|
|
||||||
if (is_bool($options['sink'])) {
|
|
||||||
throw new \InvalidArgumentException('sink must not be a boolean');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$request = Psr7\modify_request($request, $modify);
|
|
||||||
if ($request->getBody() instanceof Psr7\MultipartStream) {
|
|
||||||
// Use a multipart/form-data POST if a Content-Type is not set.
|
|
||||||
// Ensure that we don't have the header in different case and set the new value.
|
|
||||||
$options['_conditional'] = Psr7\_caseless_remove(['Content-Type'], $options['_conditional']);
|
|
||||||
$options['_conditional']['Content-Type'] = 'multipart/form-data; boundary='
|
|
||||||
. $request->getBody()->getBoundary();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Merge in conditional headers if they are not present.
|
|
||||||
if (isset($options['_conditional'])) {
|
|
||||||
// Build up the changes so it's in a single clone of the message.
|
|
||||||
$modify = [];
|
|
||||||
foreach ($options['_conditional'] as $k => $v) {
|
|
||||||
if (!$request->hasHeader($k)) {
|
|
||||||
$modify['set_headers'][$k] = $v;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$request = Psr7\modify_request($request, $modify);
|
|
||||||
// Don't pass this internal value along to middleware/handlers.
|
|
||||||
unset($options['_conditional']);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $request;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Throw Exception with pre-set message.
|
|
||||||
* @return void
|
|
||||||
* @throws \InvalidArgumentException Invalid body.
|
|
||||||
*/
|
|
||||||
private function invalidBody()
|
|
||||||
{
|
|
||||||
throw new \InvalidArgumentException('Passing in the "body" request '
|
|
||||||
. 'option as an array to send a POST request has been deprecated. '
|
|
||||||
. 'Please use the "form_params" request option to send a '
|
|
||||||
. 'application/x-www-form-urlencoded request, or the "multipart" '
|
|
||||||
. 'request option to send a multipart/form-data request.');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Exception\GuzzleException;
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Http\Message\UriInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Client interface for sending HTTP requests.
|
|
||||||
*/
|
|
||||||
interface ClientInterface
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* @deprecated Will be removed in Guzzle 7.0.0
|
|
||||||
*/
|
|
||||||
const VERSION = '6.5.5';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send an HTTP request.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request to send
|
|
||||||
* @param array $options Request options to apply to the given
|
|
||||||
* request and to the transfer.
|
|
||||||
*
|
|
||||||
* @return ResponseInterface
|
|
||||||
* @throws GuzzleException
|
|
||||||
*/
|
|
||||||
public function send(RequestInterface $request, array $options = []);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Asynchronously send an HTTP request.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request to send
|
|
||||||
* @param array $options Request options to apply to the given
|
|
||||||
* request and to the transfer.
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
public function sendAsync(RequestInterface $request, array $options = []);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create and send an HTTP request.
|
|
||||||
*
|
|
||||||
* Use an absolute path to override the base path of the client, or a
|
|
||||||
* relative path to append to the base path of the client. The URL can
|
|
||||||
* contain the query string as well.
|
|
||||||
*
|
|
||||||
* @param string $method HTTP method.
|
|
||||||
* @param string|UriInterface $uri URI object or string.
|
|
||||||
* @param array $options Request options to apply.
|
|
||||||
*
|
|
||||||
* @return ResponseInterface
|
|
||||||
* @throws GuzzleException
|
|
||||||
*/
|
|
||||||
public function request($method, $uri, array $options = []);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create and send an asynchronous HTTP request.
|
|
||||||
*
|
|
||||||
* Use an absolute path to override the base path of the client, or a
|
|
||||||
* relative path to append to the base path of the client. The URL can
|
|
||||||
* contain the query string as well. Use an array to provide a URL
|
|
||||||
* template and additional variables to use in the URL template expansion.
|
|
||||||
*
|
|
||||||
* @param string $method HTTP method
|
|
||||||
* @param string|UriInterface $uri URI object or string.
|
|
||||||
* @param array $options Request options to apply.
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
public function requestAsync($method, $uri, array $options = []);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a client configuration option.
|
|
||||||
*
|
|
||||||
* These options include default request options of the client, a "handler"
|
|
||||||
* (if utilized by the concrete client), and a "base_uri" if utilized by
|
|
||||||
* the concrete client.
|
|
||||||
*
|
|
||||||
* @param string|null $option The config option to retrieve.
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function getConfig($option = null);
|
|
||||||
}
|
|
||||||
-316
@@ -1,316 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Cookie;
|
|
||||||
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cookie jar that stores cookies as an array
|
|
||||||
*/
|
|
||||||
class CookieJar implements CookieJarInterface
|
|
||||||
{
|
|
||||||
/** @var SetCookie[] Loaded cookie data */
|
|
||||||
private $cookies = [];
|
|
||||||
|
|
||||||
/** @var bool */
|
|
||||||
private $strictMode;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param bool $strictMode Set to true to throw exceptions when invalid
|
|
||||||
* cookies are added to the cookie jar.
|
|
||||||
* @param array $cookieArray Array of SetCookie objects or a hash of
|
|
||||||
* arrays that can be used with the SetCookie
|
|
||||||
* constructor
|
|
||||||
*/
|
|
||||||
public function __construct($strictMode = false, $cookieArray = [])
|
|
||||||
{
|
|
||||||
$this->strictMode = $strictMode;
|
|
||||||
|
|
||||||
foreach ($cookieArray as $cookie) {
|
|
||||||
if (!($cookie instanceof SetCookie)) {
|
|
||||||
$cookie = new SetCookie($cookie);
|
|
||||||
}
|
|
||||||
$this->setCookie($cookie);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new Cookie jar from an associative array and domain.
|
|
||||||
*
|
|
||||||
* @param array $cookies Cookies to create the jar from
|
|
||||||
* @param string $domain Domain to set the cookies to
|
|
||||||
*
|
|
||||||
* @return self
|
|
||||||
*/
|
|
||||||
public static function fromArray(array $cookies, $domain)
|
|
||||||
{
|
|
||||||
$cookieJar = new self();
|
|
||||||
foreach ($cookies as $name => $value) {
|
|
||||||
$cookieJar->setCookie(new SetCookie([
|
|
||||||
'Domain' => $domain,
|
|
||||||
'Name' => $name,
|
|
||||||
'Value' => $value,
|
|
||||||
'Discard' => true
|
|
||||||
]));
|
|
||||||
}
|
|
||||||
|
|
||||||
return $cookieJar;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @deprecated
|
|
||||||
*/
|
|
||||||
public static function getCookieValue($value)
|
|
||||||
{
|
|
||||||
return $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Evaluate if this cookie should be persisted to storage
|
|
||||||
* that survives between requests.
|
|
||||||
*
|
|
||||||
* @param SetCookie $cookie Being evaluated.
|
|
||||||
* @param bool $allowSessionCookies If we should persist session cookies
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public static function shouldPersist(
|
|
||||||
SetCookie $cookie,
|
|
||||||
$allowSessionCookies = false
|
|
||||||
) {
|
|
||||||
if ($cookie->getExpires() || $allowSessionCookies) {
|
|
||||||
if (!$cookie->getDiscard()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Finds and returns the cookie based on the name
|
|
||||||
*
|
|
||||||
* @param string $name cookie name to search for
|
|
||||||
* @return SetCookie|null cookie that was found or null if not found
|
|
||||||
*/
|
|
||||||
public function getCookieByName($name)
|
|
||||||
{
|
|
||||||
// don't allow a non string name
|
|
||||||
if ($name === null || !is_scalar($name)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
foreach ($this->cookies as $cookie) {
|
|
||||||
if ($cookie->getName() !== null && strcasecmp($cookie->getName(), $name) === 0) {
|
|
||||||
return $cookie;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function toArray()
|
|
||||||
{
|
|
||||||
return array_map(function (SetCookie $cookie) {
|
|
||||||
return $cookie->toArray();
|
|
||||||
}, $this->getIterator()->getArrayCopy());
|
|
||||||
}
|
|
||||||
|
|
||||||
public function clear($domain = null, $path = null, $name = null)
|
|
||||||
{
|
|
||||||
if (!$domain) {
|
|
||||||
$this->cookies = [];
|
|
||||||
return;
|
|
||||||
} elseif (!$path) {
|
|
||||||
$this->cookies = array_filter(
|
|
||||||
$this->cookies,
|
|
||||||
function (SetCookie $cookie) use ($domain) {
|
|
||||||
return !$cookie->matchesDomain($domain);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
} elseif (!$name) {
|
|
||||||
$this->cookies = array_filter(
|
|
||||||
$this->cookies,
|
|
||||||
function (SetCookie $cookie) use ($path, $domain) {
|
|
||||||
return !($cookie->matchesPath($path) &&
|
|
||||||
$cookie->matchesDomain($domain));
|
|
||||||
}
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
$this->cookies = array_filter(
|
|
||||||
$this->cookies,
|
|
||||||
function (SetCookie $cookie) use ($path, $domain, $name) {
|
|
||||||
return !($cookie->getName() == $name &&
|
|
||||||
$cookie->matchesPath($path) &&
|
|
||||||
$cookie->matchesDomain($domain));
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function clearSessionCookies()
|
|
||||||
{
|
|
||||||
$this->cookies = array_filter(
|
|
||||||
$this->cookies,
|
|
||||||
function (SetCookie $cookie) {
|
|
||||||
return !$cookie->getDiscard() && $cookie->getExpires();
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function setCookie(SetCookie $cookie)
|
|
||||||
{
|
|
||||||
// If the name string is empty (but not 0), ignore the set-cookie
|
|
||||||
// string entirely.
|
|
||||||
$name = $cookie->getName();
|
|
||||||
if (!$name && $name !== '0') {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Only allow cookies with set and valid domain, name, value
|
|
||||||
$result = $cookie->validate();
|
|
||||||
if ($result !== true) {
|
|
||||||
if ($this->strictMode) {
|
|
||||||
throw new \RuntimeException('Invalid cookie: ' . $result);
|
|
||||||
} else {
|
|
||||||
$this->removeCookieIfEmpty($cookie);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Resolve conflicts with previously set cookies
|
|
||||||
foreach ($this->cookies as $i => $c) {
|
|
||||||
|
|
||||||
// Two cookies are identical, when their path, and domain are
|
|
||||||
// identical.
|
|
||||||
if ($c->getPath() != $cookie->getPath() ||
|
|
||||||
$c->getDomain() != $cookie->getDomain() ||
|
|
||||||
$c->getName() != $cookie->getName()
|
|
||||||
) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// The previously set cookie is a discard cookie and this one is
|
|
||||||
// not so allow the new cookie to be set
|
|
||||||
if (!$cookie->getDiscard() && $c->getDiscard()) {
|
|
||||||
unset($this->cookies[$i]);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the new cookie's expiration is further into the future, then
|
|
||||||
// replace the old cookie
|
|
||||||
if ($cookie->getExpires() > $c->getExpires()) {
|
|
||||||
unset($this->cookies[$i]);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the value has changed, we better change it
|
|
||||||
if ($cookie->getValue() !== $c->getValue()) {
|
|
||||||
unset($this->cookies[$i]);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// The cookie exists, so no need to continue
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->cookies[] = $cookie;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function count()
|
|
||||||
{
|
|
||||||
return count($this->cookies);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getIterator()
|
|
||||||
{
|
|
||||||
return new \ArrayIterator(array_values($this->cookies));
|
|
||||||
}
|
|
||||||
|
|
||||||
public function extractCookies(
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response
|
|
||||||
) {
|
|
||||||
if ($cookieHeader = $response->getHeader('Set-Cookie')) {
|
|
||||||
foreach ($cookieHeader as $cookie) {
|
|
||||||
$sc = SetCookie::fromString($cookie);
|
|
||||||
if (!$sc->getDomain()) {
|
|
||||||
$sc->setDomain($request->getUri()->getHost());
|
|
||||||
}
|
|
||||||
if (0 !== strpos($sc->getPath(), '/')) {
|
|
||||||
$sc->setPath($this->getCookiePathFromRequest($request));
|
|
||||||
}
|
|
||||||
$this->setCookie($sc);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Computes cookie path following RFC 6265 section 5.1.4
|
|
||||||
*
|
|
||||||
* @link https://tools.ietf.org/html/rfc6265#section-5.1.4
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
private function getCookiePathFromRequest(RequestInterface $request)
|
|
||||||
{
|
|
||||||
$uriPath = $request->getUri()->getPath();
|
|
||||||
if ('' === $uriPath) {
|
|
||||||
return '/';
|
|
||||||
}
|
|
||||||
if (0 !== strpos($uriPath, '/')) {
|
|
||||||
return '/';
|
|
||||||
}
|
|
||||||
if ('/' === $uriPath) {
|
|
||||||
return '/';
|
|
||||||
}
|
|
||||||
if (0 === $lastSlashPos = strrpos($uriPath, '/')) {
|
|
||||||
return '/';
|
|
||||||
}
|
|
||||||
|
|
||||||
return substr($uriPath, 0, $lastSlashPos);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function withCookieHeader(RequestInterface $request)
|
|
||||||
{
|
|
||||||
$values = [];
|
|
||||||
$uri = $request->getUri();
|
|
||||||
$scheme = $uri->getScheme();
|
|
||||||
$host = $uri->getHost();
|
|
||||||
$path = $uri->getPath() ?: '/';
|
|
||||||
|
|
||||||
foreach ($this->cookies as $cookie) {
|
|
||||||
if ($cookie->matchesPath($path) &&
|
|
||||||
$cookie->matchesDomain($host) &&
|
|
||||||
!$cookie->isExpired() &&
|
|
||||||
(!$cookie->getSecure() || $scheme === 'https')
|
|
||||||
) {
|
|
||||||
$values[] = $cookie->getName() . '='
|
|
||||||
. $cookie->getValue();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $values
|
|
||||||
? $request->withHeader('Cookie', implode('; ', $values))
|
|
||||||
: $request;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* If a cookie already exists and the server asks to set it again with a
|
|
||||||
* null value, the cookie must be deleted.
|
|
||||||
*
|
|
||||||
* @param SetCookie $cookie
|
|
||||||
*/
|
|
||||||
private function removeCookieIfEmpty(SetCookie $cookie)
|
|
||||||
{
|
|
||||||
$cookieValue = $cookie->getValue();
|
|
||||||
if ($cookieValue === null || $cookieValue === '') {
|
|
||||||
$this->clear(
|
|
||||||
$cookie->getDomain(),
|
|
||||||
$cookie->getPath(),
|
|
||||||
$cookie->getName()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,84 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Cookie;
|
|
||||||
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Stores HTTP cookies.
|
|
||||||
*
|
|
||||||
* It extracts cookies from HTTP requests, and returns them in HTTP responses.
|
|
||||||
* CookieJarInterface instances automatically expire contained cookies when
|
|
||||||
* necessary. Subclasses are also responsible for storing and retrieving
|
|
||||||
* cookies from a file, database, etc.
|
|
||||||
*
|
|
||||||
* @link http://docs.python.org/2/library/cookielib.html Inspiration
|
|
||||||
*/
|
|
||||||
interface CookieJarInterface extends \Countable, \IteratorAggregate
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Create a request with added cookie headers.
|
|
||||||
*
|
|
||||||
* If no matching cookies are found in the cookie jar, then no Cookie
|
|
||||||
* header is added to the request and the same request is returned.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request object to modify.
|
|
||||||
*
|
|
||||||
* @return RequestInterface returns the modified request.
|
|
||||||
*/
|
|
||||||
public function withCookieHeader(RequestInterface $request);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Extract cookies from an HTTP response and store them in the CookieJar.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request that was sent
|
|
||||||
* @param ResponseInterface $response Response that was received
|
|
||||||
*/
|
|
||||||
public function extractCookies(
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sets a cookie in the cookie jar.
|
|
||||||
*
|
|
||||||
* @param SetCookie $cookie Cookie to set.
|
|
||||||
*
|
|
||||||
* @return bool Returns true on success or false on failure
|
|
||||||
*/
|
|
||||||
public function setCookie(SetCookie $cookie);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Remove cookies currently held in the cookie jar.
|
|
||||||
*
|
|
||||||
* Invoking this method without arguments will empty the whole cookie jar.
|
|
||||||
* If given a $domain argument only cookies belonging to that domain will
|
|
||||||
* be removed. If given a $domain and $path argument, cookies belonging to
|
|
||||||
* the specified path within that domain are removed. If given all three
|
|
||||||
* arguments, then the cookie with the specified name, path and domain is
|
|
||||||
* removed.
|
|
||||||
*
|
|
||||||
* @param string|null $domain Clears cookies matching a domain
|
|
||||||
* @param string|null $path Clears cookies matching a domain and path
|
|
||||||
* @param string|null $name Clears cookies matching a domain, path, and name
|
|
||||||
*
|
|
||||||
* @return CookieJarInterface
|
|
||||||
*/
|
|
||||||
public function clear($domain = null, $path = null, $name = null);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Discard all sessions cookies.
|
|
||||||
*
|
|
||||||
* Removes cookies that don't have an expire field or a have a discard
|
|
||||||
* field set to true. To be called when the user agent shuts down according
|
|
||||||
* to RFC 2965.
|
|
||||||
*/
|
|
||||||
public function clearSessionCookies();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Converts the cookie jar to an array.
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public function toArray();
|
|
||||||
}
|
|
||||||
@@ -1,91 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Cookie;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Persists non-session cookies using a JSON formatted file
|
|
||||||
*/
|
|
||||||
class FileCookieJar extends CookieJar
|
|
||||||
{
|
|
||||||
/** @var string filename */
|
|
||||||
private $filename;
|
|
||||||
|
|
||||||
/** @var bool Control whether to persist session cookies or not. */
|
|
||||||
private $storeSessionCookies;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new FileCookieJar object
|
|
||||||
*
|
|
||||||
* @param string $cookieFile File to store the cookie data
|
|
||||||
* @param bool $storeSessionCookies Set to true to store session cookies
|
|
||||||
* in the cookie jar.
|
|
||||||
*
|
|
||||||
* @throws \RuntimeException if the file cannot be found or created
|
|
||||||
*/
|
|
||||||
public function __construct($cookieFile, $storeSessionCookies = false)
|
|
||||||
{
|
|
||||||
parent::__construct();
|
|
||||||
$this->filename = $cookieFile;
|
|
||||||
$this->storeSessionCookies = $storeSessionCookies;
|
|
||||||
|
|
||||||
if (file_exists($cookieFile)) {
|
|
||||||
$this->load($cookieFile);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Saves the file when shutting down
|
|
||||||
*/
|
|
||||||
public function __destruct()
|
|
||||||
{
|
|
||||||
$this->save($this->filename);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Saves the cookies to a file.
|
|
||||||
*
|
|
||||||
* @param string $filename File to save
|
|
||||||
* @throws \RuntimeException if the file cannot be found or created
|
|
||||||
*/
|
|
||||||
public function save($filename)
|
|
||||||
{
|
|
||||||
$json = [];
|
|
||||||
foreach ($this as $cookie) {
|
|
||||||
/** @var SetCookie $cookie */
|
|
||||||
if (CookieJar::shouldPersist($cookie, $this->storeSessionCookies)) {
|
|
||||||
$json[] = $cookie->toArray();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$jsonStr = \GuzzleHttp\json_encode($json);
|
|
||||||
if (false === file_put_contents($filename, $jsonStr, LOCK_EX)) {
|
|
||||||
throw new \RuntimeException("Unable to save file {$filename}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Load cookies from a JSON formatted file.
|
|
||||||
*
|
|
||||||
* Old cookies are kept unless overwritten by newly loaded ones.
|
|
||||||
*
|
|
||||||
* @param string $filename Cookie file to load.
|
|
||||||
* @throws \RuntimeException if the file cannot be loaded.
|
|
||||||
*/
|
|
||||||
public function load($filename)
|
|
||||||
{
|
|
||||||
$json = file_get_contents($filename);
|
|
||||||
if (false === $json) {
|
|
||||||
throw new \RuntimeException("Unable to load file {$filename}");
|
|
||||||
} elseif ($json === '') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$data = \GuzzleHttp\json_decode($json, true);
|
|
||||||
if (is_array($data)) {
|
|
||||||
foreach (json_decode($json, true) as $cookie) {
|
|
||||||
$this->setCookie(new SetCookie($cookie));
|
|
||||||
}
|
|
||||||
} elseif (strlen($data)) {
|
|
||||||
throw new \RuntimeException("Invalid cookie file: {$filename}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,72 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Cookie;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Persists cookies in the client session
|
|
||||||
*/
|
|
||||||
class SessionCookieJar extends CookieJar
|
|
||||||
{
|
|
||||||
/** @var string session key */
|
|
||||||
private $sessionKey;
|
|
||||||
|
|
||||||
/** @var bool Control whether to persist session cookies or not. */
|
|
||||||
private $storeSessionCookies;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new SessionCookieJar object
|
|
||||||
*
|
|
||||||
* @param string $sessionKey Session key name to store the cookie
|
|
||||||
* data in session
|
|
||||||
* @param bool $storeSessionCookies Set to true to store session cookies
|
|
||||||
* in the cookie jar.
|
|
||||||
*/
|
|
||||||
public function __construct($sessionKey, $storeSessionCookies = false)
|
|
||||||
{
|
|
||||||
parent::__construct();
|
|
||||||
$this->sessionKey = $sessionKey;
|
|
||||||
$this->storeSessionCookies = $storeSessionCookies;
|
|
||||||
$this->load();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Saves cookies to session when shutting down
|
|
||||||
*/
|
|
||||||
public function __destruct()
|
|
||||||
{
|
|
||||||
$this->save();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Save cookies to the client session
|
|
||||||
*/
|
|
||||||
public function save()
|
|
||||||
{
|
|
||||||
$json = [];
|
|
||||||
foreach ($this as $cookie) {
|
|
||||||
/** @var SetCookie $cookie */
|
|
||||||
if (CookieJar::shouldPersist($cookie, $this->storeSessionCookies)) {
|
|
||||||
$json[] = $cookie->toArray();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$_SESSION[$this->sessionKey] = json_encode($json);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Load the contents of the client session into the data array
|
|
||||||
*/
|
|
||||||
protected function load()
|
|
||||||
{
|
|
||||||
if (!isset($_SESSION[$this->sessionKey])) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
$data = json_decode($_SESSION[$this->sessionKey], true);
|
|
||||||
if (is_array($data)) {
|
|
||||||
foreach ($data as $cookie) {
|
|
||||||
$this->setCookie(new SetCookie($cookie));
|
|
||||||
}
|
|
||||||
} elseif (strlen($data)) {
|
|
||||||
throw new \RuntimeException("Invalid cookie data");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-403
@@ -1,403 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Cookie;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set-Cookie object
|
|
||||||
*/
|
|
||||||
class SetCookie
|
|
||||||
{
|
|
||||||
/** @var array */
|
|
||||||
private static $defaults = [
|
|
||||||
'Name' => null,
|
|
||||||
'Value' => null,
|
|
||||||
'Domain' => null,
|
|
||||||
'Path' => '/',
|
|
||||||
'Max-Age' => null,
|
|
||||||
'Expires' => null,
|
|
||||||
'Secure' => false,
|
|
||||||
'Discard' => false,
|
|
||||||
'HttpOnly' => false
|
|
||||||
];
|
|
||||||
|
|
||||||
/** @var array Cookie data */
|
|
||||||
private $data;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new SetCookie object from a string
|
|
||||||
*
|
|
||||||
* @param string $cookie Set-Cookie header string
|
|
||||||
*
|
|
||||||
* @return self
|
|
||||||
*/
|
|
||||||
public static function fromString($cookie)
|
|
||||||
{
|
|
||||||
// Create the default return array
|
|
||||||
$data = self::$defaults;
|
|
||||||
// Explode the cookie string using a series of semicolons
|
|
||||||
$pieces = array_filter(array_map('trim', explode(';', $cookie)));
|
|
||||||
// The name of the cookie (first kvp) must exist and include an equal sign.
|
|
||||||
if (empty($pieces[0]) || !strpos($pieces[0], '=')) {
|
|
||||||
return new self($data);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add the cookie pieces into the parsed data array
|
|
||||||
foreach ($pieces as $part) {
|
|
||||||
$cookieParts = explode('=', $part, 2);
|
|
||||||
$key = trim($cookieParts[0]);
|
|
||||||
$value = isset($cookieParts[1])
|
|
||||||
? trim($cookieParts[1], " \n\r\t\0\x0B")
|
|
||||||
: true;
|
|
||||||
|
|
||||||
// Only check for non-cookies when cookies have been found
|
|
||||||
if (empty($data['Name'])) {
|
|
||||||
$data['Name'] = $key;
|
|
||||||
$data['Value'] = $value;
|
|
||||||
} else {
|
|
||||||
foreach (array_keys(self::$defaults) as $search) {
|
|
||||||
if (!strcasecmp($search, $key)) {
|
|
||||||
$data[$search] = $value;
|
|
||||||
continue 2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$data[$key] = $value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return new self($data);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param array $data Array of cookie data provided by a Cookie parser
|
|
||||||
*/
|
|
||||||
public function __construct(array $data = [])
|
|
||||||
{
|
|
||||||
$this->data = array_replace(self::$defaults, $data);
|
|
||||||
// Extract the Expires value and turn it into a UNIX timestamp if needed
|
|
||||||
if (!$this->getExpires() && $this->getMaxAge()) {
|
|
||||||
// Calculate the Expires date
|
|
||||||
$this->setExpires(time() + $this->getMaxAge());
|
|
||||||
} elseif ($this->getExpires() && !is_numeric($this->getExpires())) {
|
|
||||||
$this->setExpires($this->getExpires());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __toString()
|
|
||||||
{
|
|
||||||
$str = $this->data['Name'] . '=' . $this->data['Value'] . '; ';
|
|
||||||
foreach ($this->data as $k => $v) {
|
|
||||||
if ($k !== 'Name' && $k !== 'Value' && $v !== null && $v !== false) {
|
|
||||||
if ($k === 'Expires') {
|
|
||||||
$str .= 'Expires=' . gmdate('D, d M Y H:i:s \G\M\T', $v) . '; ';
|
|
||||||
} else {
|
|
||||||
$str .= ($v === true ? $k : "{$k}={$v}") . '; ';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return rtrim($str, '; ');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function toArray()
|
|
||||||
{
|
|
||||||
return $this->data;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the cookie name
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function getName()
|
|
||||||
{
|
|
||||||
return $this->data['Name'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the cookie name
|
|
||||||
*
|
|
||||||
* @param string $name Cookie name
|
|
||||||
*/
|
|
||||||
public function setName($name)
|
|
||||||
{
|
|
||||||
$this->data['Name'] = $name;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the cookie value
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function getValue()
|
|
||||||
{
|
|
||||||
return $this->data['Value'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the cookie value
|
|
||||||
*
|
|
||||||
* @param string $value Cookie value
|
|
||||||
*/
|
|
||||||
public function setValue($value)
|
|
||||||
{
|
|
||||||
$this->data['Value'] = $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the domain
|
|
||||||
*
|
|
||||||
* @return string|null
|
|
||||||
*/
|
|
||||||
public function getDomain()
|
|
||||||
{
|
|
||||||
return $this->data['Domain'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the domain of the cookie
|
|
||||||
*
|
|
||||||
* @param string $domain
|
|
||||||
*/
|
|
||||||
public function setDomain($domain)
|
|
||||||
{
|
|
||||||
$this->data['Domain'] = $domain;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the path
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function getPath()
|
|
||||||
{
|
|
||||||
return $this->data['Path'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the path of the cookie
|
|
||||||
*
|
|
||||||
* @param string $path Path of the cookie
|
|
||||||
*/
|
|
||||||
public function setPath($path)
|
|
||||||
{
|
|
||||||
$this->data['Path'] = $path;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Maximum lifetime of the cookie in seconds
|
|
||||||
*
|
|
||||||
* @return int|null
|
|
||||||
*/
|
|
||||||
public function getMaxAge()
|
|
||||||
{
|
|
||||||
return $this->data['Max-Age'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the max-age of the cookie
|
|
||||||
*
|
|
||||||
* @param int $maxAge Max age of the cookie in seconds
|
|
||||||
*/
|
|
||||||
public function setMaxAge($maxAge)
|
|
||||||
{
|
|
||||||
$this->data['Max-Age'] = $maxAge;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The UNIX timestamp when the cookie Expires
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function getExpires()
|
|
||||||
{
|
|
||||||
return $this->data['Expires'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the unix timestamp for which the cookie will expire
|
|
||||||
*
|
|
||||||
* @param int $timestamp Unix timestamp
|
|
||||||
*/
|
|
||||||
public function setExpires($timestamp)
|
|
||||||
{
|
|
||||||
$this->data['Expires'] = is_numeric($timestamp)
|
|
||||||
? (int) $timestamp
|
|
||||||
: strtotime($timestamp);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether or not this is a secure cookie
|
|
||||||
*
|
|
||||||
* @return bool|null
|
|
||||||
*/
|
|
||||||
public function getSecure()
|
|
||||||
{
|
|
||||||
return $this->data['Secure'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether or not the cookie is secure
|
|
||||||
*
|
|
||||||
* @param bool $secure Set to true or false if secure
|
|
||||||
*/
|
|
||||||
public function setSecure($secure)
|
|
||||||
{
|
|
||||||
$this->data['Secure'] = $secure;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether or not this is a session cookie
|
|
||||||
*
|
|
||||||
* @return bool|null
|
|
||||||
*/
|
|
||||||
public function getDiscard()
|
|
||||||
{
|
|
||||||
return $this->data['Discard'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether or not this is a session cookie
|
|
||||||
*
|
|
||||||
* @param bool $discard Set to true or false if this is a session cookie
|
|
||||||
*/
|
|
||||||
public function setDiscard($discard)
|
|
||||||
{
|
|
||||||
$this->data['Discard'] = $discard;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether or not this is an HTTP only cookie
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function getHttpOnly()
|
|
||||||
{
|
|
||||||
return $this->data['HttpOnly'];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether or not this is an HTTP only cookie
|
|
||||||
*
|
|
||||||
* @param bool $httpOnly Set to true or false if this is HTTP only
|
|
||||||
*/
|
|
||||||
public function setHttpOnly($httpOnly)
|
|
||||||
{
|
|
||||||
$this->data['HttpOnly'] = $httpOnly;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if the cookie matches a path value.
|
|
||||||
*
|
|
||||||
* A request-path path-matches a given cookie-path if at least one of
|
|
||||||
* the following conditions holds:
|
|
||||||
*
|
|
||||||
* - The cookie-path and the request-path are identical.
|
|
||||||
* - The cookie-path is a prefix of the request-path, and the last
|
|
||||||
* character of the cookie-path is %x2F ("/").
|
|
||||||
* - The cookie-path is a prefix of the request-path, and the first
|
|
||||||
* character of the request-path that is not included in the cookie-
|
|
||||||
* path is a %x2F ("/") character.
|
|
||||||
*
|
|
||||||
* @param string $requestPath Path to check against
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function matchesPath($requestPath)
|
|
||||||
{
|
|
||||||
$cookiePath = $this->getPath();
|
|
||||||
|
|
||||||
// Match on exact matches or when path is the default empty "/"
|
|
||||||
if ($cookiePath === '/' || $cookiePath == $requestPath) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure that the cookie-path is a prefix of the request path.
|
|
||||||
if (0 !== strpos($requestPath, $cookiePath)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Match if the last character of the cookie-path is "/"
|
|
||||||
if (substr($cookiePath, -1, 1) === '/') {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Match if the first character not included in cookie path is "/"
|
|
||||||
return substr($requestPath, strlen($cookiePath), 1) === '/';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if the cookie matches a domain value
|
|
||||||
*
|
|
||||||
* @param string $domain Domain to check against
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function matchesDomain($domain)
|
|
||||||
{
|
|
||||||
// Remove the leading '.' as per spec in RFC 6265.
|
|
||||||
// http://tools.ietf.org/html/rfc6265#section-5.2.3
|
|
||||||
$cookieDomain = ltrim($this->getDomain(), '.');
|
|
||||||
|
|
||||||
// Domain not set or exact match.
|
|
||||||
if (!$cookieDomain || !strcasecmp($domain, $cookieDomain)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Matching the subdomain according to RFC 6265.
|
|
||||||
// http://tools.ietf.org/html/rfc6265#section-5.1.3
|
|
||||||
if (filter_var($domain, FILTER_VALIDATE_IP)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (bool) preg_match('/\.' . preg_quote($cookieDomain, '/') . '$/', $domain);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if the cookie is expired
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function isExpired()
|
|
||||||
{
|
|
||||||
return $this->getExpires() !== null && time() > $this->getExpires();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if the cookie is valid according to RFC 6265
|
|
||||||
*
|
|
||||||
* @return bool|string Returns true if valid or an error message if invalid
|
|
||||||
*/
|
|
||||||
public function validate()
|
|
||||||
{
|
|
||||||
// Names must not be empty, but can be 0
|
|
||||||
$name = $this->getName();
|
|
||||||
if (empty($name) && !is_numeric($name)) {
|
|
||||||
return 'The cookie name must not be empty';
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if any of the invalid characters are present in the cookie name
|
|
||||||
if (preg_match(
|
|
||||||
'/[\x00-\x20\x22\x28-\x29\x2c\x2f\x3a-\x40\x5c\x7b\x7d\x7f]/',
|
|
||||||
$name
|
|
||||||
)) {
|
|
||||||
return 'Cookie name must not contain invalid characters: ASCII '
|
|
||||||
. 'Control characters (0-31;127), space, tab and the '
|
|
||||||
. 'following characters: ()<>@,;:\"/?={}';
|
|
||||||
}
|
|
||||||
|
|
||||||
// Value must not be empty, but can be 0
|
|
||||||
$value = $this->getValue();
|
|
||||||
if (empty($value) && !is_numeric($value)) {
|
|
||||||
return 'The cookie value must not be empty';
|
|
||||||
}
|
|
||||||
|
|
||||||
// Domains must not be empty, but can be 0
|
|
||||||
// A "0" is not a valid internet domain, but may be used as server name
|
|
||||||
// in a private network.
|
|
||||||
$domain = $this->getDomain();
|
|
||||||
if (empty($domain) && !is_numeric($domain)) {
|
|
||||||
return 'The cookie domain must not be empty';
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exception when an HTTP error occurs (4xx or 5xx error)
|
|
||||||
*/
|
|
||||||
class BadResponseException extends RequestException
|
|
||||||
{
|
|
||||||
public function __construct(
|
|
||||||
$message,
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response = null,
|
|
||||||
\Exception $previous = null,
|
|
||||||
array $handlerContext = []
|
|
||||||
) {
|
|
||||||
if (null === $response) {
|
|
||||||
@trigger_error(
|
|
||||||
'Instantiating the ' . __CLASS__ . ' class without a Response is deprecated since version 6.3 and will be removed in 7.0.',
|
|
||||||
E_USER_DEPRECATED
|
|
||||||
);
|
|
||||||
}
|
|
||||||
parent::__construct($message, $request, $response, $previous, $handlerContext);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exception when a client error is encountered (4xx codes)
|
|
||||||
*/
|
|
||||||
class ClientException extends BadResponseException
|
|
||||||
{
|
|
||||||
}
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exception thrown when a connection cannot be established.
|
|
||||||
*
|
|
||||||
* Note that no response is present for a ConnectException
|
|
||||||
*/
|
|
||||||
class ConnectException extends RequestException
|
|
||||||
{
|
|
||||||
public function __construct(
|
|
||||||
$message,
|
|
||||||
RequestInterface $request,
|
|
||||||
\Exception $previous = null,
|
|
||||||
array $handlerContext = []
|
|
||||||
) {
|
|
||||||
parent::__construct($message, $request, null, $previous, $handlerContext);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return null
|
|
||||||
*/
|
|
||||||
public function getResponse()
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function hasResponse()
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,23 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
use Throwable;
|
|
||||||
|
|
||||||
if (interface_exists(Throwable::class)) {
|
|
||||||
interface GuzzleException extends Throwable
|
|
||||||
{
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
/**
|
|
||||||
* @method string getMessage()
|
|
||||||
* @method \Throwable|null getPrevious()
|
|
||||||
* @method mixed getCode()
|
|
||||||
* @method string getFile()
|
|
||||||
* @method int getLine()
|
|
||||||
* @method array getTrace()
|
|
||||||
* @method string getTraceAsString()
|
|
||||||
*/
|
|
||||||
interface GuzzleException
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
final class InvalidArgumentException extends \InvalidArgumentException implements GuzzleException
|
|
||||||
{
|
|
||||||
}
|
|
||||||
@@ -1,192 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Http\Message\UriInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* HTTP Request exception
|
|
||||||
*/
|
|
||||||
class RequestException extends TransferException
|
|
||||||
{
|
|
||||||
/** @var RequestInterface */
|
|
||||||
private $request;
|
|
||||||
|
|
||||||
/** @var ResponseInterface|null */
|
|
||||||
private $response;
|
|
||||||
|
|
||||||
/** @var array */
|
|
||||||
private $handlerContext;
|
|
||||||
|
|
||||||
public function __construct(
|
|
||||||
$message,
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response = null,
|
|
||||||
\Exception $previous = null,
|
|
||||||
array $handlerContext = []
|
|
||||||
) {
|
|
||||||
// Set the code of the exception if the response is set and not future.
|
|
||||||
$code = $response && !($response instanceof PromiseInterface)
|
|
||||||
? $response->getStatusCode()
|
|
||||||
: 0;
|
|
||||||
parent::__construct($message, $code, $previous);
|
|
||||||
$this->request = $request;
|
|
||||||
$this->response = $response;
|
|
||||||
$this->handlerContext = $handlerContext;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrap non-RequestExceptions with a RequestException
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param \Exception $e
|
|
||||||
*
|
|
||||||
* @return RequestException
|
|
||||||
*/
|
|
||||||
public static function wrapException(RequestInterface $request, \Exception $e)
|
|
||||||
{
|
|
||||||
return $e instanceof RequestException
|
|
||||||
? $e
|
|
||||||
: new RequestException($e->getMessage(), $request, null, $e);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Factory method to create a new exception with a normalized error message
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request
|
|
||||||
* @param ResponseInterface $response Response received
|
|
||||||
* @param \Exception $previous Previous exception
|
|
||||||
* @param array $ctx Optional handler context.
|
|
||||||
*
|
|
||||||
* @return self
|
|
||||||
*/
|
|
||||||
public static function create(
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response = null,
|
|
||||||
\Exception $previous = null,
|
|
||||||
array $ctx = []
|
|
||||||
) {
|
|
||||||
if (!$response) {
|
|
||||||
return new self(
|
|
||||||
'Error completing request',
|
|
||||||
$request,
|
|
||||||
null,
|
|
||||||
$previous,
|
|
||||||
$ctx
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
$level = (int) floor($response->getStatusCode() / 100);
|
|
||||||
if ($level === 4) {
|
|
||||||
$label = 'Client error';
|
|
||||||
$className = ClientException::class;
|
|
||||||
} elseif ($level === 5) {
|
|
||||||
$label = 'Server error';
|
|
||||||
$className = ServerException::class;
|
|
||||||
} else {
|
|
||||||
$label = 'Unsuccessful request';
|
|
||||||
$className = __CLASS__;
|
|
||||||
}
|
|
||||||
|
|
||||||
$uri = $request->getUri();
|
|
||||||
$uri = static::obfuscateUri($uri);
|
|
||||||
|
|
||||||
// Client Error: `GET /` resulted in a `404 Not Found` response:
|
|
||||||
// <html> ... (truncated)
|
|
||||||
$message = sprintf(
|
|
||||||
'%s: `%s %s` resulted in a `%s %s` response',
|
|
||||||
$label,
|
|
||||||
$request->getMethod(),
|
|
||||||
$uri,
|
|
||||||
$response->getStatusCode(),
|
|
||||||
$response->getReasonPhrase()
|
|
||||||
);
|
|
||||||
|
|
||||||
$summary = static::getResponseBodySummary($response);
|
|
||||||
|
|
||||||
if ($summary !== null) {
|
|
||||||
$message .= ":\n{$summary}\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
return new $className($message, $request, $response, $previous, $ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a short summary of the response
|
|
||||||
*
|
|
||||||
* Will return `null` if the response is not printable.
|
|
||||||
*
|
|
||||||
* @param ResponseInterface $response
|
|
||||||
*
|
|
||||||
* @return string|null
|
|
||||||
*/
|
|
||||||
public static function getResponseBodySummary(ResponseInterface $response)
|
|
||||||
{
|
|
||||||
return \GuzzleHttp\Psr7\get_message_body_summary($response);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Obfuscates URI if there is a username and a password present
|
|
||||||
*
|
|
||||||
* @param UriInterface $uri
|
|
||||||
*
|
|
||||||
* @return UriInterface
|
|
||||||
*/
|
|
||||||
private static function obfuscateUri(UriInterface $uri)
|
|
||||||
{
|
|
||||||
$userInfo = $uri->getUserInfo();
|
|
||||||
|
|
||||||
if (false !== ($pos = strpos($userInfo, ':'))) {
|
|
||||||
return $uri->withUserInfo(substr($userInfo, 0, $pos), '***');
|
|
||||||
}
|
|
||||||
|
|
||||||
return $uri;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the request that caused the exception
|
|
||||||
*
|
|
||||||
* @return RequestInterface
|
|
||||||
*/
|
|
||||||
public function getRequest()
|
|
||||||
{
|
|
||||||
return $this->request;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the associated response
|
|
||||||
*
|
|
||||||
* @return ResponseInterface|null
|
|
||||||
*/
|
|
||||||
public function getResponse()
|
|
||||||
{
|
|
||||||
return $this->response;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check if a response was received
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function hasResponse()
|
|
||||||
{
|
|
||||||
return $this->response !== null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get contextual information about the error from the underlying handler.
|
|
||||||
*
|
|
||||||
* The contents of this array will vary depending on which handler you are
|
|
||||||
* using. It may also be just an empty array. Relying on this data will
|
|
||||||
* couple you to a specific handler, but can give more debug information
|
|
||||||
* when needed.
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public function getHandlerContext()
|
|
||||||
{
|
|
||||||
return $this->handlerContext;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
use Psr\Http\Message\StreamInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exception thrown when a seek fails on a stream.
|
|
||||||
*/
|
|
||||||
class SeekException extends \RuntimeException implements GuzzleException
|
|
||||||
{
|
|
||||||
private $stream;
|
|
||||||
|
|
||||||
public function __construct(StreamInterface $stream, $pos = 0, $msg = '')
|
|
||||||
{
|
|
||||||
$this->stream = $stream;
|
|
||||||
$msg = $msg ?: 'Could not seek the stream to position ' . $pos;
|
|
||||||
parent::__construct($msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return StreamInterface
|
|
||||||
*/
|
|
||||||
public function getStream()
|
|
||||||
{
|
|
||||||
return $this->stream;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exception when a server error is encountered (5xx codes)
|
|
||||||
*/
|
|
||||||
class ServerException extends BadResponseException
|
|
||||||
{
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
class TooManyRedirectsException extends RequestException
|
|
||||||
{
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Exception;
|
|
||||||
|
|
||||||
class TransferException extends \RuntimeException implements GuzzleException
|
|
||||||
{
|
|
||||||
}
|
|
||||||
@@ -1,585 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use GuzzleHttp\Exception\ConnectException;
|
|
||||||
use GuzzleHttp\Exception\RequestException;
|
|
||||||
use GuzzleHttp\Promise\FulfilledPromise;
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use GuzzleHttp\Psr7\LazyOpenStream;
|
|
||||||
use GuzzleHttp\TransferStats;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates curl resources from a request
|
|
||||||
*/
|
|
||||||
class CurlFactory implements CurlFactoryInterface
|
|
||||||
{
|
|
||||||
const CURL_VERSION_STR = 'curl_version';
|
|
||||||
const LOW_CURL_VERSION_NUMBER = '7.21.2';
|
|
||||||
|
|
||||||
/** @var array */
|
|
||||||
private $handles = [];
|
|
||||||
|
|
||||||
/** @var int Total number of idle handles to keep in cache */
|
|
||||||
private $maxHandles;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param int $maxHandles Maximum number of idle handles.
|
|
||||||
*/
|
|
||||||
public function __construct($maxHandles)
|
|
||||||
{
|
|
||||||
$this->maxHandles = $maxHandles;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function create(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
if (isset($options['curl']['body_as_string'])) {
|
|
||||||
$options['_body_as_string'] = $options['curl']['body_as_string'];
|
|
||||||
unset($options['curl']['body_as_string']);
|
|
||||||
}
|
|
||||||
|
|
||||||
$easy = new EasyHandle;
|
|
||||||
$easy->request = $request;
|
|
||||||
$easy->options = $options;
|
|
||||||
$conf = $this->getDefaultConf($easy);
|
|
||||||
$this->applyMethod($easy, $conf);
|
|
||||||
$this->applyHandlerOptions($easy, $conf);
|
|
||||||
$this->applyHeaders($easy, $conf);
|
|
||||||
unset($conf['_headers']);
|
|
||||||
|
|
||||||
// Add handler options from the request configuration options
|
|
||||||
if (isset($options['curl'])) {
|
|
||||||
$conf = array_replace($conf, $options['curl']);
|
|
||||||
}
|
|
||||||
|
|
||||||
$conf[CURLOPT_HEADERFUNCTION] = $this->createHeaderFn($easy);
|
|
||||||
$easy->handle = $this->handles
|
|
||||||
? array_pop($this->handles)
|
|
||||||
: curl_init();
|
|
||||||
curl_setopt_array($easy->handle, $conf);
|
|
||||||
|
|
||||||
return $easy;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function release(EasyHandle $easy)
|
|
||||||
{
|
|
||||||
$resource = $easy->handle;
|
|
||||||
unset($easy->handle);
|
|
||||||
|
|
||||||
if (count($this->handles) >= $this->maxHandles) {
|
|
||||||
curl_close($resource);
|
|
||||||
} else {
|
|
||||||
// Remove all callback functions as they can hold onto references
|
|
||||||
// and are not cleaned up by curl_reset. Using curl_setopt_array
|
|
||||||
// does not work for some reason, so removing each one
|
|
||||||
// individually.
|
|
||||||
curl_setopt($resource, CURLOPT_HEADERFUNCTION, null);
|
|
||||||
curl_setopt($resource, CURLOPT_READFUNCTION, null);
|
|
||||||
curl_setopt($resource, CURLOPT_WRITEFUNCTION, null);
|
|
||||||
curl_setopt($resource, CURLOPT_PROGRESSFUNCTION, null);
|
|
||||||
curl_reset($resource);
|
|
||||||
$this->handles[] = $resource;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Completes a cURL transaction, either returning a response promise or a
|
|
||||||
* rejected promise.
|
|
||||||
*
|
|
||||||
* @param callable $handler
|
|
||||||
* @param EasyHandle $easy
|
|
||||||
* @param CurlFactoryInterface $factory Dictates how the handle is released
|
|
||||||
*
|
|
||||||
* @return \GuzzleHttp\Promise\PromiseInterface
|
|
||||||
*/
|
|
||||||
public static function finish(
|
|
||||||
callable $handler,
|
|
||||||
EasyHandle $easy,
|
|
||||||
CurlFactoryInterface $factory
|
|
||||||
) {
|
|
||||||
if (isset($easy->options['on_stats'])) {
|
|
||||||
self::invokeStats($easy);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!$easy->response || $easy->errno) {
|
|
||||||
return self::finishError($handler, $easy, $factory);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return the response if it is present and there is no error.
|
|
||||||
$factory->release($easy);
|
|
||||||
|
|
||||||
// Rewind the body of the response if possible.
|
|
||||||
$body = $easy->response->getBody();
|
|
||||||
if ($body->isSeekable()) {
|
|
||||||
$body->rewind();
|
|
||||||
}
|
|
||||||
|
|
||||||
return new FulfilledPromise($easy->response);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static function invokeStats(EasyHandle $easy)
|
|
||||||
{
|
|
||||||
$curlStats = curl_getinfo($easy->handle);
|
|
||||||
$curlStats['appconnect_time'] = curl_getinfo($easy->handle, CURLINFO_APPCONNECT_TIME);
|
|
||||||
$stats = new TransferStats(
|
|
||||||
$easy->request,
|
|
||||||
$easy->response,
|
|
||||||
$curlStats['total_time'],
|
|
||||||
$easy->errno,
|
|
||||||
$curlStats
|
|
||||||
);
|
|
||||||
call_user_func($easy->options['on_stats'], $stats);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static function finishError(
|
|
||||||
callable $handler,
|
|
||||||
EasyHandle $easy,
|
|
||||||
CurlFactoryInterface $factory
|
|
||||||
) {
|
|
||||||
// Get error information and release the handle to the factory.
|
|
||||||
$ctx = [
|
|
||||||
'errno' => $easy->errno,
|
|
||||||
'error' => curl_error($easy->handle),
|
|
||||||
'appconnect_time' => curl_getinfo($easy->handle, CURLINFO_APPCONNECT_TIME),
|
|
||||||
] + curl_getinfo($easy->handle);
|
|
||||||
$ctx[self::CURL_VERSION_STR] = curl_version()['version'];
|
|
||||||
$factory->release($easy);
|
|
||||||
|
|
||||||
// Retry when nothing is present or when curl failed to rewind.
|
|
||||||
if (empty($easy->options['_err_message'])
|
|
||||||
&& (!$easy->errno || $easy->errno == 65)
|
|
||||||
) {
|
|
||||||
return self::retryFailedRewind($handler, $easy, $ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
return self::createRejection($easy, $ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static function createRejection(EasyHandle $easy, array $ctx)
|
|
||||||
{
|
|
||||||
static $connectionErrors = [
|
|
||||||
CURLE_OPERATION_TIMEOUTED => true,
|
|
||||||
CURLE_COULDNT_RESOLVE_HOST => true,
|
|
||||||
CURLE_COULDNT_CONNECT => true,
|
|
||||||
CURLE_SSL_CONNECT_ERROR => true,
|
|
||||||
CURLE_GOT_NOTHING => true,
|
|
||||||
];
|
|
||||||
|
|
||||||
// If an exception was encountered during the onHeaders event, then
|
|
||||||
// return a rejected promise that wraps that exception.
|
|
||||||
if ($easy->onHeadersException) {
|
|
||||||
return \GuzzleHttp\Promise\rejection_for(
|
|
||||||
new RequestException(
|
|
||||||
'An error was encountered during the on_headers event',
|
|
||||||
$easy->request,
|
|
||||||
$easy->response,
|
|
||||||
$easy->onHeadersException,
|
|
||||||
$ctx
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (version_compare($ctx[self::CURL_VERSION_STR], self::LOW_CURL_VERSION_NUMBER)) {
|
|
||||||
$message = sprintf(
|
|
||||||
'cURL error %s: %s (%s)',
|
|
||||||
$ctx['errno'],
|
|
||||||
$ctx['error'],
|
|
||||||
'see https://curl.haxx.se/libcurl/c/libcurl-errors.html'
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
$message = sprintf(
|
|
||||||
'cURL error %s: %s (%s) for %s',
|
|
||||||
$ctx['errno'],
|
|
||||||
$ctx['error'],
|
|
||||||
'see https://curl.haxx.se/libcurl/c/libcurl-errors.html',
|
|
||||||
$easy->request->getUri()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create a connection exception if it was a specific error code.
|
|
||||||
$error = isset($connectionErrors[$easy->errno])
|
|
||||||
? new ConnectException($message, $easy->request, null, $ctx)
|
|
||||||
: new RequestException($message, $easy->request, $easy->response, null, $ctx);
|
|
||||||
|
|
||||||
return \GuzzleHttp\Promise\rejection_for($error);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function getDefaultConf(EasyHandle $easy)
|
|
||||||
{
|
|
||||||
$conf = [
|
|
||||||
'_headers' => $easy->request->getHeaders(),
|
|
||||||
CURLOPT_CUSTOMREQUEST => $easy->request->getMethod(),
|
|
||||||
CURLOPT_URL => (string) $easy->request->getUri()->withFragment(''),
|
|
||||||
CURLOPT_RETURNTRANSFER => false,
|
|
||||||
CURLOPT_HEADER => false,
|
|
||||||
CURLOPT_CONNECTTIMEOUT => 150,
|
|
||||||
];
|
|
||||||
|
|
||||||
if (defined('CURLOPT_PROTOCOLS')) {
|
|
||||||
$conf[CURLOPT_PROTOCOLS] = CURLPROTO_HTTP | CURLPROTO_HTTPS;
|
|
||||||
}
|
|
||||||
|
|
||||||
$version = $easy->request->getProtocolVersion();
|
|
||||||
if ($version == 1.1) {
|
|
||||||
$conf[CURLOPT_HTTP_VERSION] = CURL_HTTP_VERSION_1_1;
|
|
||||||
} elseif ($version == 2.0) {
|
|
||||||
$conf[CURLOPT_HTTP_VERSION] = CURL_HTTP_VERSION_2_0;
|
|
||||||
} else {
|
|
||||||
$conf[CURLOPT_HTTP_VERSION] = CURL_HTTP_VERSION_1_0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $conf;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function applyMethod(EasyHandle $easy, array &$conf)
|
|
||||||
{
|
|
||||||
$body = $easy->request->getBody();
|
|
||||||
$size = $body->getSize();
|
|
||||||
|
|
||||||
if ($size === null || $size > 0) {
|
|
||||||
$this->applyBody($easy->request, $easy->options, $conf);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$method = $easy->request->getMethod();
|
|
||||||
if ($method === 'PUT' || $method === 'POST') {
|
|
||||||
// See http://tools.ietf.org/html/rfc7230#section-3.3.2
|
|
||||||
if (!$easy->request->hasHeader('Content-Length')) {
|
|
||||||
$conf[CURLOPT_HTTPHEADER][] = 'Content-Length: 0';
|
|
||||||
}
|
|
||||||
} elseif ($method === 'HEAD') {
|
|
||||||
$conf[CURLOPT_NOBODY] = true;
|
|
||||||
unset(
|
|
||||||
$conf[CURLOPT_WRITEFUNCTION],
|
|
||||||
$conf[CURLOPT_READFUNCTION],
|
|
||||||
$conf[CURLOPT_FILE],
|
|
||||||
$conf[CURLOPT_INFILE]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function applyBody(RequestInterface $request, array $options, array &$conf)
|
|
||||||
{
|
|
||||||
$size = $request->hasHeader('Content-Length')
|
|
||||||
? (int) $request->getHeaderLine('Content-Length')
|
|
||||||
: null;
|
|
||||||
|
|
||||||
// Send the body as a string if the size is less than 1MB OR if the
|
|
||||||
// [curl][body_as_string] request value is set.
|
|
||||||
if (($size !== null && $size < 1000000) ||
|
|
||||||
!empty($options['_body_as_string'])
|
|
||||||
) {
|
|
||||||
$conf[CURLOPT_POSTFIELDS] = (string) $request->getBody();
|
|
||||||
// Don't duplicate the Content-Length header
|
|
||||||
$this->removeHeader('Content-Length', $conf);
|
|
||||||
$this->removeHeader('Transfer-Encoding', $conf);
|
|
||||||
} else {
|
|
||||||
$conf[CURLOPT_UPLOAD] = true;
|
|
||||||
if ($size !== null) {
|
|
||||||
$conf[CURLOPT_INFILESIZE] = $size;
|
|
||||||
$this->removeHeader('Content-Length', $conf);
|
|
||||||
}
|
|
||||||
$body = $request->getBody();
|
|
||||||
if ($body->isSeekable()) {
|
|
||||||
$body->rewind();
|
|
||||||
}
|
|
||||||
$conf[CURLOPT_READFUNCTION] = function ($ch, $fd, $length) use ($body) {
|
|
||||||
return $body->read($length);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the Expect header is not present, prevent curl from adding it
|
|
||||||
if (!$request->hasHeader('Expect')) {
|
|
||||||
$conf[CURLOPT_HTTPHEADER][] = 'Expect:';
|
|
||||||
}
|
|
||||||
|
|
||||||
// cURL sometimes adds a content-type by default. Prevent this.
|
|
||||||
if (!$request->hasHeader('Content-Type')) {
|
|
||||||
$conf[CURLOPT_HTTPHEADER][] = 'Content-Type:';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function applyHeaders(EasyHandle $easy, array &$conf)
|
|
||||||
{
|
|
||||||
foreach ($conf['_headers'] as $name => $values) {
|
|
||||||
foreach ($values as $value) {
|
|
||||||
$value = (string) $value;
|
|
||||||
if ($value === '') {
|
|
||||||
// cURL requires a special format for empty headers.
|
|
||||||
// See https://github.com/guzzle/guzzle/issues/1882 for more details.
|
|
||||||
$conf[CURLOPT_HTTPHEADER][] = "$name;";
|
|
||||||
} else {
|
|
||||||
$conf[CURLOPT_HTTPHEADER][] = "$name: $value";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove the Accept header if one was not set
|
|
||||||
if (!$easy->request->hasHeader('Accept')) {
|
|
||||||
$conf[CURLOPT_HTTPHEADER][] = 'Accept:';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Remove a header from the options array.
|
|
||||||
*
|
|
||||||
* @param string $name Case-insensitive header to remove
|
|
||||||
* @param array $options Array of options to modify
|
|
||||||
*/
|
|
||||||
private function removeHeader($name, array &$options)
|
|
||||||
{
|
|
||||||
foreach (array_keys($options['_headers']) as $key) {
|
|
||||||
if (!strcasecmp($key, $name)) {
|
|
||||||
unset($options['_headers'][$key]);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function applyHandlerOptions(EasyHandle $easy, array &$conf)
|
|
||||||
{
|
|
||||||
$options = $easy->options;
|
|
||||||
if (isset($options['verify'])) {
|
|
||||||
if ($options['verify'] === false) {
|
|
||||||
unset($conf[CURLOPT_CAINFO]);
|
|
||||||
$conf[CURLOPT_SSL_VERIFYHOST] = 0;
|
|
||||||
$conf[CURLOPT_SSL_VERIFYPEER] = false;
|
|
||||||
} else {
|
|
||||||
$conf[CURLOPT_SSL_VERIFYHOST] = 2;
|
|
||||||
$conf[CURLOPT_SSL_VERIFYPEER] = true;
|
|
||||||
if (is_string($options['verify'])) {
|
|
||||||
// Throw an error if the file/folder/link path is not valid or doesn't exist.
|
|
||||||
if (!file_exists($options['verify'])) {
|
|
||||||
throw new \InvalidArgumentException(
|
|
||||||
"SSL CA bundle not found: {$options['verify']}"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
// If it's a directory or a link to a directory use CURLOPT_CAPATH.
|
|
||||||
// If not, it's probably a file, or a link to a file, so use CURLOPT_CAINFO.
|
|
||||||
if (is_dir($options['verify']) ||
|
|
||||||
(is_link($options['verify']) && is_dir(readlink($options['verify'])))) {
|
|
||||||
$conf[CURLOPT_CAPATH] = $options['verify'];
|
|
||||||
} else {
|
|
||||||
$conf[CURLOPT_CAINFO] = $options['verify'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($options['decode_content'])) {
|
|
||||||
$accept = $easy->request->getHeaderLine('Accept-Encoding');
|
|
||||||
if ($accept) {
|
|
||||||
$conf[CURLOPT_ENCODING] = $accept;
|
|
||||||
} else {
|
|
||||||
$conf[CURLOPT_ENCODING] = '';
|
|
||||||
// Don't let curl send the header over the wire
|
|
||||||
$conf[CURLOPT_HTTPHEADER][] = 'Accept-Encoding:';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['sink'])) {
|
|
||||||
$sink = $options['sink'];
|
|
||||||
if (!is_string($sink)) {
|
|
||||||
$sink = \GuzzleHttp\Psr7\stream_for($sink);
|
|
||||||
} elseif (!is_dir(dirname($sink))) {
|
|
||||||
// Ensure that the directory exists before failing in curl.
|
|
||||||
throw new \RuntimeException(sprintf(
|
|
||||||
'Directory %s does not exist for sink value of %s',
|
|
||||||
dirname($sink),
|
|
||||||
$sink
|
|
||||||
));
|
|
||||||
} else {
|
|
||||||
$sink = new LazyOpenStream($sink, 'w+');
|
|
||||||
}
|
|
||||||
$easy->sink = $sink;
|
|
||||||
$conf[CURLOPT_WRITEFUNCTION] = function ($ch, $write) use ($sink) {
|
|
||||||
return $sink->write($write);
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
// Use a default temp stream if no sink was set.
|
|
||||||
$conf[CURLOPT_FILE] = fopen('php://temp', 'w+');
|
|
||||||
$easy->sink = Psr7\stream_for($conf[CURLOPT_FILE]);
|
|
||||||
}
|
|
||||||
$timeoutRequiresNoSignal = false;
|
|
||||||
if (isset($options['timeout'])) {
|
|
||||||
$timeoutRequiresNoSignal |= $options['timeout'] < 1;
|
|
||||||
$conf[CURLOPT_TIMEOUT_MS] = $options['timeout'] * 1000;
|
|
||||||
}
|
|
||||||
|
|
||||||
// CURL default value is CURL_IPRESOLVE_WHATEVER
|
|
||||||
if (isset($options['force_ip_resolve'])) {
|
|
||||||
if ('v4' === $options['force_ip_resolve']) {
|
|
||||||
$conf[CURLOPT_IPRESOLVE] = CURL_IPRESOLVE_V4;
|
|
||||||
} elseif ('v6' === $options['force_ip_resolve']) {
|
|
||||||
$conf[CURLOPT_IPRESOLVE] = CURL_IPRESOLVE_V6;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['connect_timeout'])) {
|
|
||||||
$timeoutRequiresNoSignal |= $options['connect_timeout'] < 1;
|
|
||||||
$conf[CURLOPT_CONNECTTIMEOUT_MS] = $options['connect_timeout'] * 1000;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($timeoutRequiresNoSignal && strtoupper(substr(PHP_OS, 0, 3)) !== 'WIN') {
|
|
||||||
$conf[CURLOPT_NOSIGNAL] = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['proxy'])) {
|
|
||||||
if (!is_array($options['proxy'])) {
|
|
||||||
$conf[CURLOPT_PROXY] = $options['proxy'];
|
|
||||||
} else {
|
|
||||||
$scheme = $easy->request->getUri()->getScheme();
|
|
||||||
if (isset($options['proxy'][$scheme])) {
|
|
||||||
$host = $easy->request->getUri()->getHost();
|
|
||||||
if (!isset($options['proxy']['no']) ||
|
|
||||||
!\GuzzleHttp\is_host_in_noproxy($host, $options['proxy']['no'])
|
|
||||||
) {
|
|
||||||
$conf[CURLOPT_PROXY] = $options['proxy'][$scheme];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['cert'])) {
|
|
||||||
$cert = $options['cert'];
|
|
||||||
if (is_array($cert)) {
|
|
||||||
$conf[CURLOPT_SSLCERTPASSWD] = $cert[1];
|
|
||||||
$cert = $cert[0];
|
|
||||||
}
|
|
||||||
if (!file_exists($cert)) {
|
|
||||||
throw new \InvalidArgumentException(
|
|
||||||
"SSL certificate not found: {$cert}"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
$conf[CURLOPT_SSLCERT] = $cert;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['ssl_key'])) {
|
|
||||||
if (is_array($options['ssl_key'])) {
|
|
||||||
if (count($options['ssl_key']) === 2) {
|
|
||||||
list($sslKey, $conf[CURLOPT_SSLKEYPASSWD]) = $options['ssl_key'];
|
|
||||||
} else {
|
|
||||||
list($sslKey) = $options['ssl_key'];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$sslKey = isset($sslKey) ? $sslKey: $options['ssl_key'];
|
|
||||||
|
|
||||||
if (!file_exists($sslKey)) {
|
|
||||||
throw new \InvalidArgumentException(
|
|
||||||
"SSL private key not found: {$sslKey}"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
$conf[CURLOPT_SSLKEY] = $sslKey;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['progress'])) {
|
|
||||||
$progress = $options['progress'];
|
|
||||||
if (!is_callable($progress)) {
|
|
||||||
throw new \InvalidArgumentException(
|
|
||||||
'progress client option must be callable'
|
|
||||||
);
|
|
||||||
}
|
|
||||||
$conf[CURLOPT_NOPROGRESS] = false;
|
|
||||||
$conf[CURLOPT_PROGRESSFUNCTION] = function () use ($progress) {
|
|
||||||
$args = func_get_args();
|
|
||||||
// PHP 5.5 pushed the handle onto the start of the args
|
|
||||||
if (is_resource($args[0])) {
|
|
||||||
array_shift($args);
|
|
||||||
}
|
|
||||||
call_user_func_array($progress, $args);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($options['debug'])) {
|
|
||||||
$conf[CURLOPT_STDERR] = \GuzzleHttp\debug_resource($options['debug']);
|
|
||||||
$conf[CURLOPT_VERBOSE] = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function ensures that a response was set on a transaction. If one
|
|
||||||
* was not set, then the request is retried if possible. This error
|
|
||||||
* typically means you are sending a payload, curl encountered a
|
|
||||||
* "Connection died, retrying a fresh connect" error, tried to rewind the
|
|
||||||
* stream, and then encountered a "necessary data rewind wasn't possible"
|
|
||||||
* error, causing the request to be sent through curl_multi_info_read()
|
|
||||||
* without an error status.
|
|
||||||
*/
|
|
||||||
private static function retryFailedRewind(
|
|
||||||
callable $handler,
|
|
||||||
EasyHandle $easy,
|
|
||||||
array $ctx
|
|
||||||
) {
|
|
||||||
try {
|
|
||||||
// Only rewind if the body has been read from.
|
|
||||||
$body = $easy->request->getBody();
|
|
||||||
if ($body->tell() > 0) {
|
|
||||||
$body->rewind();
|
|
||||||
}
|
|
||||||
} catch (\RuntimeException $e) {
|
|
||||||
$ctx['error'] = 'The connection unexpectedly failed without '
|
|
||||||
. 'providing an error. The request would have been retried, '
|
|
||||||
. 'but attempting to rewind the request body failed. '
|
|
||||||
. 'Exception: ' . $e;
|
|
||||||
return self::createRejection($easy, $ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Retry no more than 3 times before giving up.
|
|
||||||
if (!isset($easy->options['_curl_retries'])) {
|
|
||||||
$easy->options['_curl_retries'] = 1;
|
|
||||||
} elseif ($easy->options['_curl_retries'] == 2) {
|
|
||||||
$ctx['error'] = 'The cURL request was retried 3 times '
|
|
||||||
. 'and did not succeed. The most likely reason for the failure '
|
|
||||||
. 'is that cURL was unable to rewind the body of the request '
|
|
||||||
. 'and subsequent retries resulted in the same error. Turn on '
|
|
||||||
. 'the debug option to see what went wrong. See '
|
|
||||||
. 'https://bugs.php.net/bug.php?id=47204 for more information.';
|
|
||||||
return self::createRejection($easy, $ctx);
|
|
||||||
} else {
|
|
||||||
$easy->options['_curl_retries']++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $handler($easy->request, $easy->options);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function createHeaderFn(EasyHandle $easy)
|
|
||||||
{
|
|
||||||
if (isset($easy->options['on_headers'])) {
|
|
||||||
$onHeaders = $easy->options['on_headers'];
|
|
||||||
|
|
||||||
if (!is_callable($onHeaders)) {
|
|
||||||
throw new \InvalidArgumentException('on_headers must be callable');
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
$onHeaders = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return function ($ch, $h) use (
|
|
||||||
$onHeaders,
|
|
||||||
$easy,
|
|
||||||
&$startingResponse
|
|
||||||
) {
|
|
||||||
$value = trim($h);
|
|
||||||
if ($value === '') {
|
|
||||||
$startingResponse = true;
|
|
||||||
$easy->createResponse();
|
|
||||||
if ($onHeaders !== null) {
|
|
||||||
try {
|
|
||||||
$onHeaders($easy->response);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
// Associate the exception with the handle and trigger
|
|
||||||
// a curl header write error by returning 0.
|
|
||||||
$easy->onHeadersException = $e;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} elseif ($startingResponse) {
|
|
||||||
$startingResponse = false;
|
|
||||||
$easy->headers = [$value];
|
|
||||||
} else {
|
|
||||||
$easy->headers[] = $value;
|
|
||||||
}
|
|
||||||
return strlen($h);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
interface CurlFactoryInterface
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Creates a cURL handle resource.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request
|
|
||||||
* @param array $options Transfer options
|
|
||||||
*
|
|
||||||
* @return EasyHandle
|
|
||||||
* @throws \RuntimeException when an option cannot be applied
|
|
||||||
*/
|
|
||||||
public function create(RequestInterface $request, array $options);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Release an easy handle, allowing it to be reused or closed.
|
|
||||||
*
|
|
||||||
* This function must call unset on the easy handle's "handle" property.
|
|
||||||
*
|
|
||||||
* @param EasyHandle $easy
|
|
||||||
*/
|
|
||||||
public function release(EasyHandle $easy);
|
|
||||||
}
|
|
||||||
@@ -1,45 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* HTTP handler that uses cURL easy handles as a transport layer.
|
|
||||||
*
|
|
||||||
* When using the CurlHandler, custom curl options can be specified as an
|
|
||||||
* associative array of curl option constants mapping to values in the
|
|
||||||
* **curl** key of the "client" key of the request.
|
|
||||||
*/
|
|
||||||
class CurlHandler
|
|
||||||
{
|
|
||||||
/** @var CurlFactoryInterface */
|
|
||||||
private $factory;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Accepts an associative array of options:
|
|
||||||
*
|
|
||||||
* - factory: Optional curl factory used to create cURL handles.
|
|
||||||
*
|
|
||||||
* @param array $options Array of options to use with the handler
|
|
||||||
*/
|
|
||||||
public function __construct(array $options = [])
|
|
||||||
{
|
|
||||||
$this->factory = isset($options['handle_factory'])
|
|
||||||
? $options['handle_factory']
|
|
||||||
: new CurlFactory(3);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
if (isset($options['delay'])) {
|
|
||||||
usleep($options['delay'] * 1000);
|
|
||||||
}
|
|
||||||
|
|
||||||
$easy = $this->factory->create($request, $options);
|
|
||||||
curl_exec($easy->handle);
|
|
||||||
$easy->errno = curl_errno($easy->handle);
|
|
||||||
|
|
||||||
return CurlFactory::finish($this, $easy, $this->factory);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,219 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use GuzzleHttp\Promise as P;
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
use GuzzleHttp\Utils;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns an asynchronous response using curl_multi_* functions.
|
|
||||||
*
|
|
||||||
* When using the CurlMultiHandler, custom curl options can be specified as an
|
|
||||||
* associative array of curl option constants mapping to values in the
|
|
||||||
* **curl** key of the provided request options.
|
|
||||||
*
|
|
||||||
* @property resource $_mh Internal use only. Lazy loaded multi-handle.
|
|
||||||
*/
|
|
||||||
class CurlMultiHandler
|
|
||||||
{
|
|
||||||
/** @var CurlFactoryInterface */
|
|
||||||
private $factory;
|
|
||||||
private $selectTimeout;
|
|
||||||
private $active;
|
|
||||||
private $handles = [];
|
|
||||||
private $delays = [];
|
|
||||||
private $options = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This handler accepts the following options:
|
|
||||||
*
|
|
||||||
* - handle_factory: An optional factory used to create curl handles
|
|
||||||
* - select_timeout: Optional timeout (in seconds) to block before timing
|
|
||||||
* out while selecting curl handles. Defaults to 1 second.
|
|
||||||
* - options: An associative array of CURLMOPT_* options and
|
|
||||||
* corresponding values for curl_multi_setopt()
|
|
||||||
*
|
|
||||||
* @param array $options
|
|
||||||
*/
|
|
||||||
public function __construct(array $options = [])
|
|
||||||
{
|
|
||||||
$this->factory = isset($options['handle_factory'])
|
|
||||||
? $options['handle_factory'] : new CurlFactory(50);
|
|
||||||
|
|
||||||
if (isset($options['select_timeout'])) {
|
|
||||||
$this->selectTimeout = $options['select_timeout'];
|
|
||||||
} elseif ($selectTimeout = getenv('GUZZLE_CURL_SELECT_TIMEOUT')) {
|
|
||||||
$this->selectTimeout = $selectTimeout;
|
|
||||||
} else {
|
|
||||||
$this->selectTimeout = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->options = isset($options['options']) ? $options['options'] : [];
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __get($name)
|
|
||||||
{
|
|
||||||
if ($name === '_mh') {
|
|
||||||
$this->_mh = curl_multi_init();
|
|
||||||
|
|
||||||
foreach ($this->options as $option => $value) {
|
|
||||||
// A warning is raised in case of a wrong option.
|
|
||||||
curl_multi_setopt($this->_mh, $option, $value);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Further calls to _mh will return the value directly, without entering the
|
|
||||||
// __get() method at all.
|
|
||||||
return $this->_mh;
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new \BadMethodCallException();
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __destruct()
|
|
||||||
{
|
|
||||||
if (isset($this->_mh)) {
|
|
||||||
curl_multi_close($this->_mh);
|
|
||||||
unset($this->_mh);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
$easy = $this->factory->create($request, $options);
|
|
||||||
$id = (int) $easy->handle;
|
|
||||||
|
|
||||||
$promise = new Promise(
|
|
||||||
[$this, 'execute'],
|
|
||||||
function () use ($id) {
|
|
||||||
return $this->cancel($id);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
$this->addRequest(['easy' => $easy, 'deferred' => $promise]);
|
|
||||||
|
|
||||||
return $promise;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Ticks the curl event loop.
|
|
||||||
*/
|
|
||||||
public function tick()
|
|
||||||
{
|
|
||||||
// Add any delayed handles if needed.
|
|
||||||
if ($this->delays) {
|
|
||||||
$currentTime = Utils::currentTime();
|
|
||||||
foreach ($this->delays as $id => $delay) {
|
|
||||||
if ($currentTime >= $delay) {
|
|
||||||
unset($this->delays[$id]);
|
|
||||||
curl_multi_add_handle(
|
|
||||||
$this->_mh,
|
|
||||||
$this->handles[$id]['easy']->handle
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Step through the task queue which may add additional requests.
|
|
||||||
P\queue()->run();
|
|
||||||
|
|
||||||
if ($this->active &&
|
|
||||||
curl_multi_select($this->_mh, $this->selectTimeout) === -1
|
|
||||||
) {
|
|
||||||
// Perform a usleep if a select returns -1.
|
|
||||||
// See: https://bugs.php.net/bug.php?id=61141
|
|
||||||
usleep(250);
|
|
||||||
}
|
|
||||||
|
|
||||||
while (curl_multi_exec($this->_mh, $this->active) === CURLM_CALL_MULTI_PERFORM);
|
|
||||||
|
|
||||||
$this->processMessages();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Runs until all outstanding connections have completed.
|
|
||||||
*/
|
|
||||||
public function execute()
|
|
||||||
{
|
|
||||||
$queue = P\queue();
|
|
||||||
|
|
||||||
while ($this->handles || !$queue->isEmpty()) {
|
|
||||||
// If there are no transfers, then sleep for the next delay
|
|
||||||
if (!$this->active && $this->delays) {
|
|
||||||
usleep($this->timeToNext());
|
|
||||||
}
|
|
||||||
$this->tick();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function addRequest(array $entry)
|
|
||||||
{
|
|
||||||
$easy = $entry['easy'];
|
|
||||||
$id = (int) $easy->handle;
|
|
||||||
$this->handles[$id] = $entry;
|
|
||||||
if (empty($easy->options['delay'])) {
|
|
||||||
curl_multi_add_handle($this->_mh, $easy->handle);
|
|
||||||
} else {
|
|
||||||
$this->delays[$id] = Utils::currentTime() + ($easy->options['delay'] / 1000);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cancels a handle from sending and removes references to it.
|
|
||||||
*
|
|
||||||
* @param int $id Handle ID to cancel and remove.
|
|
||||||
*
|
|
||||||
* @return bool True on success, false on failure.
|
|
||||||
*/
|
|
||||||
private function cancel($id)
|
|
||||||
{
|
|
||||||
// Cannot cancel if it has been processed.
|
|
||||||
if (!isset($this->handles[$id])) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$handle = $this->handles[$id]['easy']->handle;
|
|
||||||
unset($this->delays[$id], $this->handles[$id]);
|
|
||||||
curl_multi_remove_handle($this->_mh, $handle);
|
|
||||||
curl_close($handle);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function processMessages()
|
|
||||||
{
|
|
||||||
while ($done = curl_multi_info_read($this->_mh)) {
|
|
||||||
$id = (int) $done['handle'];
|
|
||||||
curl_multi_remove_handle($this->_mh, $done['handle']);
|
|
||||||
|
|
||||||
if (!isset($this->handles[$id])) {
|
|
||||||
// Probably was cancelled.
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$entry = $this->handles[$id];
|
|
||||||
unset($this->handles[$id], $this->delays[$id]);
|
|
||||||
$entry['easy']->errno = $done['result'];
|
|
||||||
$entry['deferred']->resolve(
|
|
||||||
CurlFactory::finish(
|
|
||||||
$this,
|
|
||||||
$entry['easy'],
|
|
||||||
$this->factory
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function timeToNext()
|
|
||||||
{
|
|
||||||
$currentTime = Utils::currentTime();
|
|
||||||
$nextTime = PHP_INT_MAX;
|
|
||||||
foreach ($this->delays as $time) {
|
|
||||||
if ($time < $nextTime) {
|
|
||||||
$nextTime = $time;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return max(0, $nextTime - $currentTime) * 1000000;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,92 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use GuzzleHttp\Psr7\Response;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Http\Message\StreamInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Represents a cURL easy handle and the data it populates.
|
|
||||||
*
|
|
||||||
* @internal
|
|
||||||
*/
|
|
||||||
final class EasyHandle
|
|
||||||
{
|
|
||||||
/** @var resource cURL resource */
|
|
||||||
public $handle;
|
|
||||||
|
|
||||||
/** @var StreamInterface Where data is being written */
|
|
||||||
public $sink;
|
|
||||||
|
|
||||||
/** @var array Received HTTP headers so far */
|
|
||||||
public $headers = [];
|
|
||||||
|
|
||||||
/** @var ResponseInterface Received response (if any) */
|
|
||||||
public $response;
|
|
||||||
|
|
||||||
/** @var RequestInterface Request being sent */
|
|
||||||
public $request;
|
|
||||||
|
|
||||||
/** @var array Request options */
|
|
||||||
public $options = [];
|
|
||||||
|
|
||||||
/** @var int cURL error number (if any) */
|
|
||||||
public $errno = 0;
|
|
||||||
|
|
||||||
/** @var \Exception Exception during on_headers (if any) */
|
|
||||||
public $onHeadersException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Attach a response to the easy handle based on the received headers.
|
|
||||||
*
|
|
||||||
* @throws \RuntimeException if no headers have been received.
|
|
||||||
*/
|
|
||||||
public function createResponse()
|
|
||||||
{
|
|
||||||
if (empty($this->headers)) {
|
|
||||||
throw new \RuntimeException('No headers have been received');
|
|
||||||
}
|
|
||||||
|
|
||||||
// HTTP-version SP status-code SP reason-phrase
|
|
||||||
$startLine = explode(' ', array_shift($this->headers), 3);
|
|
||||||
$headers = \GuzzleHttp\headers_from_lines($this->headers);
|
|
||||||
$normalizedKeys = \GuzzleHttp\normalize_header_keys($headers);
|
|
||||||
|
|
||||||
if (!empty($this->options['decode_content'])
|
|
||||||
&& isset($normalizedKeys['content-encoding'])
|
|
||||||
) {
|
|
||||||
$headers['x-encoded-content-encoding']
|
|
||||||
= $headers[$normalizedKeys['content-encoding']];
|
|
||||||
unset($headers[$normalizedKeys['content-encoding']]);
|
|
||||||
if (isset($normalizedKeys['content-length'])) {
|
|
||||||
$headers['x-encoded-content-length']
|
|
||||||
= $headers[$normalizedKeys['content-length']];
|
|
||||||
|
|
||||||
$bodyLength = (int) $this->sink->getSize();
|
|
||||||
if ($bodyLength) {
|
|
||||||
$headers[$normalizedKeys['content-length']] = $bodyLength;
|
|
||||||
} else {
|
|
||||||
unset($headers[$normalizedKeys['content-length']]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Attach a response to the easy handle with the parsed headers.
|
|
||||||
$this->response = new Response(
|
|
||||||
$startLine[1],
|
|
||||||
$headers,
|
|
||||||
$this->sink,
|
|
||||||
substr($startLine[0], 5),
|
|
||||||
isset($startLine[2]) ? (string) $startLine[2] : null
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __get($name)
|
|
||||||
{
|
|
||||||
$msg = $name === 'handle'
|
|
||||||
? 'The EasyHandle has been released'
|
|
||||||
: 'Invalid property: ' . $name;
|
|
||||||
throw new \BadMethodCallException($msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,195 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use GuzzleHttp\Exception\RequestException;
|
|
||||||
use GuzzleHttp\HandlerStack;
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use GuzzleHttp\Promise\RejectedPromise;
|
|
||||||
use GuzzleHttp\TransferStats;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handler that returns responses or throw exceptions from a queue.
|
|
||||||
*/
|
|
||||||
class MockHandler implements \Countable
|
|
||||||
{
|
|
||||||
private $queue = [];
|
|
||||||
private $lastRequest;
|
|
||||||
private $lastOptions;
|
|
||||||
private $onFulfilled;
|
|
||||||
private $onRejected;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a new MockHandler that uses the default handler stack list of
|
|
||||||
* middlewares.
|
|
||||||
*
|
|
||||||
* @param array $queue Array of responses, callables, or exceptions.
|
|
||||||
* @param callable $onFulfilled Callback to invoke when the return value is fulfilled.
|
|
||||||
* @param callable $onRejected Callback to invoke when the return value is rejected.
|
|
||||||
*
|
|
||||||
* @return HandlerStack
|
|
||||||
*/
|
|
||||||
public static function createWithMiddleware(
|
|
||||||
array $queue = null,
|
|
||||||
callable $onFulfilled = null,
|
|
||||||
callable $onRejected = null
|
|
||||||
) {
|
|
||||||
return HandlerStack::create(new self($queue, $onFulfilled, $onRejected));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The passed in value must be an array of
|
|
||||||
* {@see Psr7\Http\Message\ResponseInterface} objects, Exceptions,
|
|
||||||
* callables, or Promises.
|
|
||||||
*
|
|
||||||
* @param array $queue
|
|
||||||
* @param callable $onFulfilled Callback to invoke when the return value is fulfilled.
|
|
||||||
* @param callable $onRejected Callback to invoke when the return value is rejected.
|
|
||||||
*/
|
|
||||||
public function __construct(
|
|
||||||
array $queue = null,
|
|
||||||
callable $onFulfilled = null,
|
|
||||||
callable $onRejected = null
|
|
||||||
) {
|
|
||||||
$this->onFulfilled = $onFulfilled;
|
|
||||||
$this->onRejected = $onRejected;
|
|
||||||
|
|
||||||
if ($queue) {
|
|
||||||
call_user_func_array([$this, 'append'], $queue);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
if (!$this->queue) {
|
|
||||||
throw new \OutOfBoundsException('Mock queue is empty');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['delay']) && is_numeric($options['delay'])) {
|
|
||||||
usleep($options['delay'] * 1000);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->lastRequest = $request;
|
|
||||||
$this->lastOptions = $options;
|
|
||||||
$response = array_shift($this->queue);
|
|
||||||
|
|
||||||
if (isset($options['on_headers'])) {
|
|
||||||
if (!is_callable($options['on_headers'])) {
|
|
||||||
throw new \InvalidArgumentException('on_headers must be callable');
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
$options['on_headers']($response);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
$msg = 'An error was encountered during the on_headers event';
|
|
||||||
$response = new RequestException($msg, $request, $response, $e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_callable($response)) {
|
|
||||||
$response = call_user_func($response, $request, $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
$response = $response instanceof \Exception
|
|
||||||
? \GuzzleHttp\Promise\rejection_for($response)
|
|
||||||
: \GuzzleHttp\Promise\promise_for($response);
|
|
||||||
|
|
||||||
return $response->then(
|
|
||||||
function ($value) use ($request, $options) {
|
|
||||||
$this->invokeStats($request, $options, $value);
|
|
||||||
if ($this->onFulfilled) {
|
|
||||||
call_user_func($this->onFulfilled, $value);
|
|
||||||
}
|
|
||||||
if (isset($options['sink'])) {
|
|
||||||
$contents = (string) $value->getBody();
|
|
||||||
$sink = $options['sink'];
|
|
||||||
|
|
||||||
if (is_resource($sink)) {
|
|
||||||
fwrite($sink, $contents);
|
|
||||||
} elseif (is_string($sink)) {
|
|
||||||
file_put_contents($sink, $contents);
|
|
||||||
} elseif ($sink instanceof \Psr\Http\Message\StreamInterface) {
|
|
||||||
$sink->write($contents);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $value;
|
|
||||||
},
|
|
||||||
function ($reason) use ($request, $options) {
|
|
||||||
$this->invokeStats($request, $options, null, $reason);
|
|
||||||
if ($this->onRejected) {
|
|
||||||
call_user_func($this->onRejected, $reason);
|
|
||||||
}
|
|
||||||
return \GuzzleHttp\Promise\rejection_for($reason);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Adds one or more variadic requests, exceptions, callables, or promises
|
|
||||||
* to the queue.
|
|
||||||
*/
|
|
||||||
public function append()
|
|
||||||
{
|
|
||||||
foreach (func_get_args() as $value) {
|
|
||||||
if ($value instanceof ResponseInterface
|
|
||||||
|| $value instanceof \Exception
|
|
||||||
|| $value instanceof PromiseInterface
|
|
||||||
|| is_callable($value)
|
|
||||||
) {
|
|
||||||
$this->queue[] = $value;
|
|
||||||
} else {
|
|
||||||
throw new \InvalidArgumentException('Expected a response or '
|
|
||||||
. 'exception. Found ' . \GuzzleHttp\describe_type($value));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the last received request.
|
|
||||||
*
|
|
||||||
* @return RequestInterface
|
|
||||||
*/
|
|
||||||
public function getLastRequest()
|
|
||||||
{
|
|
||||||
return $this->lastRequest;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the last received request options.
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public function getLastOptions()
|
|
||||||
{
|
|
||||||
return $this->lastOptions;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the number of remaining items in the queue.
|
|
||||||
*
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
public function count()
|
|
||||||
{
|
|
||||||
return count($this->queue);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function reset()
|
|
||||||
{
|
|
||||||
$this->queue = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
private function invokeStats(
|
|
||||||
RequestInterface $request,
|
|
||||||
array $options,
|
|
||||||
ResponseInterface $response = null,
|
|
||||||
$reason = null
|
|
||||||
) {
|
|
||||||
if (isset($options['on_stats'])) {
|
|
||||||
$transferTime = isset($options['transfer_time']) ? $options['transfer_time'] : 0;
|
|
||||||
$stats = new TransferStats($request, $response, $transferTime, $reason);
|
|
||||||
call_user_func($options['on_stats'], $stats);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-55
@@ -1,55 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use GuzzleHttp\RequestOptions;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Provides basic proxies for handlers.
|
|
||||||
*/
|
|
||||||
class Proxy
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Sends synchronous requests to a specific handler while sending all other
|
|
||||||
* requests to another handler.
|
|
||||||
*
|
|
||||||
* @param callable $default Handler used for normal responses
|
|
||||||
* @param callable $sync Handler used for synchronous responses.
|
|
||||||
*
|
|
||||||
* @return callable Returns the composed handler.
|
|
||||||
*/
|
|
||||||
public static function wrapSync(
|
|
||||||
callable $default,
|
|
||||||
callable $sync
|
|
||||||
) {
|
|
||||||
return function (RequestInterface $request, array $options) use ($default, $sync) {
|
|
||||||
return empty($options[RequestOptions::SYNCHRONOUS])
|
|
||||||
? $default($request, $options)
|
|
||||||
: $sync($request, $options);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sends streaming requests to a streaming compatible handler while sending
|
|
||||||
* all other requests to a default handler.
|
|
||||||
*
|
|
||||||
* This, for example, could be useful for taking advantage of the
|
|
||||||
* performance benefits of curl while still supporting true streaming
|
|
||||||
* through the StreamHandler.
|
|
||||||
*
|
|
||||||
* @param callable $default Handler used for non-streaming responses
|
|
||||||
* @param callable $streaming Handler used for streaming responses
|
|
||||||
*
|
|
||||||
* @return callable Returns the composed handler.
|
|
||||||
*/
|
|
||||||
public static function wrapStreaming(
|
|
||||||
callable $default,
|
|
||||||
callable $streaming
|
|
||||||
) {
|
|
||||||
return function (RequestInterface $request, array $options) use ($default, $streaming) {
|
|
||||||
return empty($options['stream'])
|
|
||||||
? $default($request, $options)
|
|
||||||
: $streaming($request, $options);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,545 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Handler;
|
|
||||||
|
|
||||||
use GuzzleHttp\Exception\ConnectException;
|
|
||||||
use GuzzleHttp\Exception\RequestException;
|
|
||||||
use GuzzleHttp\Promise\FulfilledPromise;
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use GuzzleHttp\TransferStats;
|
|
||||||
use GuzzleHttp\Utils;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Http\Message\StreamInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* HTTP handler that uses PHP's HTTP stream wrapper.
|
|
||||||
*/
|
|
||||||
class StreamHandler
|
|
||||||
{
|
|
||||||
private $lastHeaders = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sends an HTTP request.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request to send.
|
|
||||||
* @param array $options Request transfer options.
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
// Sleep if there is a delay specified.
|
|
||||||
if (isset($options['delay'])) {
|
|
||||||
usleep($options['delay'] * 1000);
|
|
||||||
}
|
|
||||||
|
|
||||||
$startTime = isset($options['on_stats']) ? Utils::currentTime() : null;
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Does not support the expect header.
|
|
||||||
$request = $request->withoutHeader('Expect');
|
|
||||||
|
|
||||||
// Append a content-length header if body size is zero to match
|
|
||||||
// cURL's behavior.
|
|
||||||
if (0 === $request->getBody()->getSize()) {
|
|
||||||
$request = $request->withHeader('Content-Length', '0');
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->createResponse(
|
|
||||||
$request,
|
|
||||||
$options,
|
|
||||||
$this->createStream($request, $options),
|
|
||||||
$startTime
|
|
||||||
);
|
|
||||||
} catch (\InvalidArgumentException $e) {
|
|
||||||
throw $e;
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
// Determine if the error was a networking error.
|
|
||||||
$message = $e->getMessage();
|
|
||||||
// This list can probably get more comprehensive.
|
|
||||||
if (strpos($message, 'getaddrinfo') // DNS lookup failed
|
|
||||||
|| strpos($message, 'Connection refused')
|
|
||||||
|| strpos($message, "couldn't connect to host") // error on HHVM
|
|
||||||
|| strpos($message, "connection attempt failed")
|
|
||||||
) {
|
|
||||||
$e = new ConnectException($e->getMessage(), $request, $e);
|
|
||||||
}
|
|
||||||
$e = RequestException::wrapException($request, $e);
|
|
||||||
$this->invokeStats($options, $request, $startTime, null, $e);
|
|
||||||
|
|
||||||
return \GuzzleHttp\Promise\rejection_for($e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function invokeStats(
|
|
||||||
array $options,
|
|
||||||
RequestInterface $request,
|
|
||||||
$startTime,
|
|
||||||
ResponseInterface $response = null,
|
|
||||||
$error = null
|
|
||||||
) {
|
|
||||||
if (isset($options['on_stats'])) {
|
|
||||||
$stats = new TransferStats(
|
|
||||||
$request,
|
|
||||||
$response,
|
|
||||||
Utils::currentTime() - $startTime,
|
|
||||||
$error,
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
call_user_func($options['on_stats'], $stats);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function createResponse(
|
|
||||||
RequestInterface $request,
|
|
||||||
array $options,
|
|
||||||
$stream,
|
|
||||||
$startTime
|
|
||||||
) {
|
|
||||||
$hdrs = $this->lastHeaders;
|
|
||||||
$this->lastHeaders = [];
|
|
||||||
$parts = explode(' ', array_shift($hdrs), 3);
|
|
||||||
$ver = explode('/', $parts[0])[1];
|
|
||||||
$status = $parts[1];
|
|
||||||
$reason = isset($parts[2]) ? $parts[2] : null;
|
|
||||||
$headers = \GuzzleHttp\headers_from_lines($hdrs);
|
|
||||||
list($stream, $headers) = $this->checkDecode($options, $headers, $stream);
|
|
||||||
$stream = Psr7\stream_for($stream);
|
|
||||||
$sink = $stream;
|
|
||||||
|
|
||||||
if (strcasecmp('HEAD', $request->getMethod())) {
|
|
||||||
$sink = $this->createSink($stream, $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
$response = new Psr7\Response($status, $headers, $sink, $ver, $reason);
|
|
||||||
|
|
||||||
if (isset($options['on_headers'])) {
|
|
||||||
try {
|
|
||||||
$options['on_headers']($response);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
$msg = 'An error was encountered during the on_headers event';
|
|
||||||
$ex = new RequestException($msg, $request, $response, $e);
|
|
||||||
return \GuzzleHttp\Promise\rejection_for($ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Do not drain when the request is a HEAD request because they have
|
|
||||||
// no body.
|
|
||||||
if ($sink !== $stream) {
|
|
||||||
$this->drain(
|
|
||||||
$stream,
|
|
||||||
$sink,
|
|
||||||
$response->getHeaderLine('Content-Length')
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->invokeStats($options, $request, $startTime, $response, null);
|
|
||||||
|
|
||||||
return new FulfilledPromise($response);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function createSink(StreamInterface $stream, array $options)
|
|
||||||
{
|
|
||||||
if (!empty($options['stream'])) {
|
|
||||||
return $stream;
|
|
||||||
}
|
|
||||||
|
|
||||||
$sink = isset($options['sink'])
|
|
||||||
? $options['sink']
|
|
||||||
: fopen('php://temp', 'r+');
|
|
||||||
|
|
||||||
return is_string($sink)
|
|
||||||
? new Psr7\LazyOpenStream($sink, 'w+')
|
|
||||||
: Psr7\stream_for($sink);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function checkDecode(array $options, array $headers, $stream)
|
|
||||||
{
|
|
||||||
// Automatically decode responses when instructed.
|
|
||||||
if (!empty($options['decode_content'])) {
|
|
||||||
$normalizedKeys = \GuzzleHttp\normalize_header_keys($headers);
|
|
||||||
if (isset($normalizedKeys['content-encoding'])) {
|
|
||||||
$encoding = $headers[$normalizedKeys['content-encoding']];
|
|
||||||
if ($encoding[0] === 'gzip' || $encoding[0] === 'deflate') {
|
|
||||||
$stream = new Psr7\InflateStream(
|
|
||||||
Psr7\stream_for($stream)
|
|
||||||
);
|
|
||||||
$headers['x-encoded-content-encoding']
|
|
||||||
= $headers[$normalizedKeys['content-encoding']];
|
|
||||||
// Remove content-encoding header
|
|
||||||
unset($headers[$normalizedKeys['content-encoding']]);
|
|
||||||
// Fix content-length header
|
|
||||||
if (isset($normalizedKeys['content-length'])) {
|
|
||||||
$headers['x-encoded-content-length']
|
|
||||||
= $headers[$normalizedKeys['content-length']];
|
|
||||||
|
|
||||||
$length = (int) $stream->getSize();
|
|
||||||
if ($length === 0) {
|
|
||||||
unset($headers[$normalizedKeys['content-length']]);
|
|
||||||
} else {
|
|
||||||
$headers[$normalizedKeys['content-length']] = [$length];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return [$stream, $headers];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Drains the source stream into the "sink" client option.
|
|
||||||
*
|
|
||||||
* @param StreamInterface $source
|
|
||||||
* @param StreamInterface $sink
|
|
||||||
* @param string $contentLength Header specifying the amount of
|
|
||||||
* data to read.
|
|
||||||
*
|
|
||||||
* @return StreamInterface
|
|
||||||
* @throws \RuntimeException when the sink option is invalid.
|
|
||||||
*/
|
|
||||||
private function drain(
|
|
||||||
StreamInterface $source,
|
|
||||||
StreamInterface $sink,
|
|
||||||
$contentLength
|
|
||||||
) {
|
|
||||||
// If a content-length header is provided, then stop reading once
|
|
||||||
// that number of bytes has been read. This can prevent infinitely
|
|
||||||
// reading from a stream when dealing with servers that do not honor
|
|
||||||
// Connection: Close headers.
|
|
||||||
Psr7\copy_to_stream(
|
|
||||||
$source,
|
|
||||||
$sink,
|
|
||||||
(strlen($contentLength) > 0 && (int) $contentLength > 0) ? (int) $contentLength : -1
|
|
||||||
);
|
|
||||||
|
|
||||||
$sink->seek(0);
|
|
||||||
$source->close();
|
|
||||||
|
|
||||||
return $sink;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a resource and check to ensure it was created successfully
|
|
||||||
*
|
|
||||||
* @param callable $callback Callable that returns stream resource
|
|
||||||
*
|
|
||||||
* @return resource
|
|
||||||
* @throws \RuntimeException on error
|
|
||||||
*/
|
|
||||||
private function createResource(callable $callback)
|
|
||||||
{
|
|
||||||
$errors = null;
|
|
||||||
set_error_handler(function ($_, $msg, $file, $line) use (&$errors) {
|
|
||||||
$errors[] = [
|
|
||||||
'message' => $msg,
|
|
||||||
'file' => $file,
|
|
||||||
'line' => $line
|
|
||||||
];
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
$resource = $callback();
|
|
||||||
restore_error_handler();
|
|
||||||
|
|
||||||
if (!$resource) {
|
|
||||||
$message = 'Error creating resource: ';
|
|
||||||
foreach ($errors as $err) {
|
|
||||||
foreach ($err as $key => $value) {
|
|
||||||
$message .= "[$key] $value" . PHP_EOL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new \RuntimeException(trim($message));
|
|
||||||
}
|
|
||||||
|
|
||||||
return $resource;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function createStream(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
static $methods;
|
|
||||||
if (!$methods) {
|
|
||||||
$methods = array_flip(get_class_methods(__CLASS__));
|
|
||||||
}
|
|
||||||
|
|
||||||
// HTTP/1.1 streams using the PHP stream wrapper require a
|
|
||||||
// Connection: close header
|
|
||||||
if ($request->getProtocolVersion() == '1.1'
|
|
||||||
&& !$request->hasHeader('Connection')
|
|
||||||
) {
|
|
||||||
$request = $request->withHeader('Connection', 'close');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ensure SSL is verified by default
|
|
||||||
if (!isset($options['verify'])) {
|
|
||||||
$options['verify'] = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
$params = [];
|
|
||||||
$context = $this->getDefaultContext($request);
|
|
||||||
|
|
||||||
if (isset($options['on_headers']) && !is_callable($options['on_headers'])) {
|
|
||||||
throw new \InvalidArgumentException('on_headers must be callable');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($options)) {
|
|
||||||
foreach ($options as $key => $value) {
|
|
||||||
$method = "add_{$key}";
|
|
||||||
if (isset($methods[$method])) {
|
|
||||||
$this->{$method}($request, $context, $value, $params);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['stream_context'])) {
|
|
||||||
if (!is_array($options['stream_context'])) {
|
|
||||||
throw new \InvalidArgumentException('stream_context must be an array');
|
|
||||||
}
|
|
||||||
$context = array_replace_recursive(
|
|
||||||
$context,
|
|
||||||
$options['stream_context']
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Microsoft NTLM authentication only supported with curl handler
|
|
||||||
if (isset($options['auth'])
|
|
||||||
&& is_array($options['auth'])
|
|
||||||
&& isset($options['auth'][2])
|
|
||||||
&& 'ntlm' == $options['auth'][2]
|
|
||||||
) {
|
|
||||||
throw new \InvalidArgumentException('Microsoft NTLM authentication only supported with curl handler');
|
|
||||||
}
|
|
||||||
|
|
||||||
$uri = $this->resolveHost($request, $options);
|
|
||||||
|
|
||||||
$context = $this->createResource(
|
|
||||||
function () use ($context, $params) {
|
|
||||||
return stream_context_create($context, $params);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
return $this->createResource(
|
|
||||||
function () use ($uri, &$http_response_header, $context, $options) {
|
|
||||||
$resource = fopen((string) $uri, 'r', null, $context);
|
|
||||||
$this->lastHeaders = $http_response_header;
|
|
||||||
|
|
||||||
if (isset($options['read_timeout'])) {
|
|
||||||
$readTimeout = $options['read_timeout'];
|
|
||||||
$sec = (int) $readTimeout;
|
|
||||||
$usec = ($readTimeout - $sec) * 100000;
|
|
||||||
stream_set_timeout($resource, $sec, $usec);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $resource;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function resolveHost(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
$uri = $request->getUri();
|
|
||||||
|
|
||||||
if (isset($options['force_ip_resolve']) && !filter_var($uri->getHost(), FILTER_VALIDATE_IP)) {
|
|
||||||
if ('v4' === $options['force_ip_resolve']) {
|
|
||||||
$records = dns_get_record($uri->getHost(), DNS_A);
|
|
||||||
if (!isset($records[0]['ip'])) {
|
|
||||||
throw new ConnectException(
|
|
||||||
sprintf(
|
|
||||||
"Could not resolve IPv4 address for host '%s'",
|
|
||||||
$uri->getHost()
|
|
||||||
),
|
|
||||||
$request
|
|
||||||
);
|
|
||||||
}
|
|
||||||
$uri = $uri->withHost($records[0]['ip']);
|
|
||||||
} elseif ('v6' === $options['force_ip_resolve']) {
|
|
||||||
$records = dns_get_record($uri->getHost(), DNS_AAAA);
|
|
||||||
if (!isset($records[0]['ipv6'])) {
|
|
||||||
throw new ConnectException(
|
|
||||||
sprintf(
|
|
||||||
"Could not resolve IPv6 address for host '%s'",
|
|
||||||
$uri->getHost()
|
|
||||||
),
|
|
||||||
$request
|
|
||||||
);
|
|
||||||
}
|
|
||||||
$uri = $uri->withHost('[' . $records[0]['ipv6'] . ']');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $uri;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function getDefaultContext(RequestInterface $request)
|
|
||||||
{
|
|
||||||
$headers = '';
|
|
||||||
foreach ($request->getHeaders() as $name => $value) {
|
|
||||||
foreach ($value as $val) {
|
|
||||||
$headers .= "$name: $val\r\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$context = [
|
|
||||||
'http' => [
|
|
||||||
'method' => $request->getMethod(),
|
|
||||||
'header' => $headers,
|
|
||||||
'protocol_version' => $request->getProtocolVersion(),
|
|
||||||
'ignore_errors' => true,
|
|
||||||
'follow_location' => 0,
|
|
||||||
],
|
|
||||||
];
|
|
||||||
|
|
||||||
$body = (string) $request->getBody();
|
|
||||||
|
|
||||||
if (!empty($body)) {
|
|
||||||
$context['http']['content'] = $body;
|
|
||||||
// Prevent the HTTP handler from adding a Content-Type header.
|
|
||||||
if (!$request->hasHeader('Content-Type')) {
|
|
||||||
$context['http']['header'] .= "Content-Type:\r\n";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$context['http']['header'] = rtrim($context['http']['header']);
|
|
||||||
|
|
||||||
return $context;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function add_proxy(RequestInterface $request, &$options, $value, &$params)
|
|
||||||
{
|
|
||||||
if (!is_array($value)) {
|
|
||||||
$options['http']['proxy'] = $value;
|
|
||||||
} else {
|
|
||||||
$scheme = $request->getUri()->getScheme();
|
|
||||||
if (isset($value[$scheme])) {
|
|
||||||
if (!isset($value['no'])
|
|
||||||
|| !\GuzzleHttp\is_host_in_noproxy(
|
|
||||||
$request->getUri()->getHost(),
|
|
||||||
$value['no']
|
|
||||||
)
|
|
||||||
) {
|
|
||||||
$options['http']['proxy'] = $value[$scheme];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function add_timeout(RequestInterface $request, &$options, $value, &$params)
|
|
||||||
{
|
|
||||||
if ($value > 0) {
|
|
||||||
$options['http']['timeout'] = $value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function add_verify(RequestInterface $request, &$options, $value, &$params)
|
|
||||||
{
|
|
||||||
if ($value === true) {
|
|
||||||
// PHP 5.6 or greater will find the system cert by default. When
|
|
||||||
// < 5.6, use the Guzzle bundled cacert.
|
|
||||||
if (PHP_VERSION_ID < 50600) {
|
|
||||||
$options['ssl']['cafile'] = \GuzzleHttp\default_ca_bundle();
|
|
||||||
}
|
|
||||||
} elseif (is_string($value)) {
|
|
||||||
$options['ssl']['cafile'] = $value;
|
|
||||||
if (!file_exists($value)) {
|
|
||||||
throw new \RuntimeException("SSL CA bundle not found: $value");
|
|
||||||
}
|
|
||||||
} elseif ($value === false) {
|
|
||||||
$options['ssl']['verify_peer'] = false;
|
|
||||||
$options['ssl']['verify_peer_name'] = false;
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
throw new \InvalidArgumentException('Invalid verify request option');
|
|
||||||
}
|
|
||||||
|
|
||||||
$options['ssl']['verify_peer'] = true;
|
|
||||||
$options['ssl']['verify_peer_name'] = true;
|
|
||||||
$options['ssl']['allow_self_signed'] = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function add_cert(RequestInterface $request, &$options, $value, &$params)
|
|
||||||
{
|
|
||||||
if (is_array($value)) {
|
|
||||||
$options['ssl']['passphrase'] = $value[1];
|
|
||||||
$value = $value[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!file_exists($value)) {
|
|
||||||
throw new \RuntimeException("SSL certificate not found: {$value}");
|
|
||||||
}
|
|
||||||
|
|
||||||
$options['ssl']['local_cert'] = $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function add_progress(RequestInterface $request, &$options, $value, &$params)
|
|
||||||
{
|
|
||||||
$this->addNotification(
|
|
||||||
$params,
|
|
||||||
function ($code, $a, $b, $c, $transferred, $total) use ($value) {
|
|
||||||
if ($code == STREAM_NOTIFY_PROGRESS) {
|
|
||||||
$value($total, $transferred, null, null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function add_debug(RequestInterface $request, &$options, $value, &$params)
|
|
||||||
{
|
|
||||||
if ($value === false) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
static $map = [
|
|
||||||
STREAM_NOTIFY_CONNECT => 'CONNECT',
|
|
||||||
STREAM_NOTIFY_AUTH_REQUIRED => 'AUTH_REQUIRED',
|
|
||||||
STREAM_NOTIFY_AUTH_RESULT => 'AUTH_RESULT',
|
|
||||||
STREAM_NOTIFY_MIME_TYPE_IS => 'MIME_TYPE_IS',
|
|
||||||
STREAM_NOTIFY_FILE_SIZE_IS => 'FILE_SIZE_IS',
|
|
||||||
STREAM_NOTIFY_REDIRECTED => 'REDIRECTED',
|
|
||||||
STREAM_NOTIFY_PROGRESS => 'PROGRESS',
|
|
||||||
STREAM_NOTIFY_FAILURE => 'FAILURE',
|
|
||||||
STREAM_NOTIFY_COMPLETED => 'COMPLETED',
|
|
||||||
STREAM_NOTIFY_RESOLVE => 'RESOLVE',
|
|
||||||
];
|
|
||||||
static $args = ['severity', 'message', 'message_code',
|
|
||||||
'bytes_transferred', 'bytes_max'];
|
|
||||||
|
|
||||||
$value = \GuzzleHttp\debug_resource($value);
|
|
||||||
$ident = $request->getMethod() . ' ' . $request->getUri()->withFragment('');
|
|
||||||
$this->addNotification(
|
|
||||||
$params,
|
|
||||||
function () use ($ident, $value, $map, $args) {
|
|
||||||
$passed = func_get_args();
|
|
||||||
$code = array_shift($passed);
|
|
||||||
fprintf($value, '<%s> [%s] ', $ident, $map[$code]);
|
|
||||||
foreach (array_filter($passed) as $i => $v) {
|
|
||||||
fwrite($value, $args[$i] . ': "' . $v . '" ');
|
|
||||||
}
|
|
||||||
fwrite($value, "\n");
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private function addNotification(array &$params, callable $notify)
|
|
||||||
{
|
|
||||||
// Wrap the existing function if needed.
|
|
||||||
if (!isset($params['notification'])) {
|
|
||||||
$params['notification'] = $notify;
|
|
||||||
} else {
|
|
||||||
$params['notification'] = $this->callArray([
|
|
||||||
$params['notification'],
|
|
||||||
$notify
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function callArray(array $functions)
|
|
||||||
{
|
|
||||||
return function () use ($functions) {
|
|
||||||
$args = func_get_args();
|
|
||||||
foreach ($functions as $fn) {
|
|
||||||
call_user_func_array($fn, $args);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-277
@@ -1,277 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a composed Guzzle handler function by stacking middlewares on top of
|
|
||||||
* an HTTP handler function.
|
|
||||||
*/
|
|
||||||
class HandlerStack
|
|
||||||
{
|
|
||||||
/** @var callable|null */
|
|
||||||
private $handler;
|
|
||||||
|
|
||||||
/** @var array */
|
|
||||||
private $stack = [];
|
|
||||||
|
|
||||||
/** @var callable|null */
|
|
||||||
private $cached;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a default handler stack that can be used by clients.
|
|
||||||
*
|
|
||||||
* The returned handler will wrap the provided handler or use the most
|
|
||||||
* appropriate default handler for your system. The returned HandlerStack has
|
|
||||||
* support for cookies, redirects, HTTP error exceptions, and preparing a body
|
|
||||||
* before sending.
|
|
||||||
*
|
|
||||||
* The returned handler stack can be passed to a client in the "handler"
|
|
||||||
* option.
|
|
||||||
*
|
|
||||||
* @param callable $handler HTTP handler function to use with the stack. If no
|
|
||||||
* handler is provided, the best handler for your
|
|
||||||
* system will be utilized.
|
|
||||||
*
|
|
||||||
* @return HandlerStack
|
|
||||||
*/
|
|
||||||
public static function create(callable $handler = null)
|
|
||||||
{
|
|
||||||
$stack = new self($handler ?: choose_handler());
|
|
||||||
$stack->push(Middleware::httpErrors(), 'http_errors');
|
|
||||||
$stack->push(Middleware::redirect(), 'allow_redirects');
|
|
||||||
$stack->push(Middleware::cookies(), 'cookies');
|
|
||||||
$stack->push(Middleware::prepareBody(), 'prepare_body');
|
|
||||||
|
|
||||||
return $stack;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param callable $handler Underlying HTTP handler.
|
|
||||||
*/
|
|
||||||
public function __construct(callable $handler = null)
|
|
||||||
{
|
|
||||||
$this->handler = $handler;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Invokes the handler stack as a composed handler
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param array $options
|
|
||||||
*
|
|
||||||
* @return ResponseInterface|PromiseInterface
|
|
||||||
*/
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
$handler = $this->resolve();
|
|
||||||
|
|
||||||
return $handler($request, $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Dumps a string representation of the stack.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function __toString()
|
|
||||||
{
|
|
||||||
$depth = 0;
|
|
||||||
$stack = [];
|
|
||||||
if ($this->handler) {
|
|
||||||
$stack[] = "0) Handler: " . $this->debugCallable($this->handler);
|
|
||||||
}
|
|
||||||
|
|
||||||
$result = '';
|
|
||||||
foreach (array_reverse($this->stack) as $tuple) {
|
|
||||||
$depth++;
|
|
||||||
$str = "{$depth}) Name: '{$tuple[1]}', ";
|
|
||||||
$str .= "Function: " . $this->debugCallable($tuple[0]);
|
|
||||||
$result = "> {$str}\n{$result}";
|
|
||||||
$stack[] = $str;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (array_keys($stack) as $k) {
|
|
||||||
$result .= "< {$stack[$k]}\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the HTTP handler that actually returns a promise.
|
|
||||||
*
|
|
||||||
* @param callable $handler Accepts a request and array of options and
|
|
||||||
* returns a Promise.
|
|
||||||
*/
|
|
||||||
public function setHandler(callable $handler)
|
|
||||||
{
|
|
||||||
$this->handler = $handler;
|
|
||||||
$this->cached = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns true if the builder has a handler.
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function hasHandler()
|
|
||||||
{
|
|
||||||
return (bool) $this->handler;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Unshift a middleware to the bottom of the stack.
|
|
||||||
*
|
|
||||||
* @param callable $middleware Middleware function
|
|
||||||
* @param string $name Name to register for this middleware.
|
|
||||||
*/
|
|
||||||
public function unshift(callable $middleware, $name = null)
|
|
||||||
{
|
|
||||||
array_unshift($this->stack, [$middleware, $name]);
|
|
||||||
$this->cached = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Push a middleware to the top of the stack.
|
|
||||||
*
|
|
||||||
* @param callable $middleware Middleware function
|
|
||||||
* @param string $name Name to register for this middleware.
|
|
||||||
*/
|
|
||||||
public function push(callable $middleware, $name = '')
|
|
||||||
{
|
|
||||||
$this->stack[] = [$middleware, $name];
|
|
||||||
$this->cached = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add a middleware before another middleware by name.
|
|
||||||
*
|
|
||||||
* @param string $findName Middleware to find
|
|
||||||
* @param callable $middleware Middleware function
|
|
||||||
* @param string $withName Name to register for this middleware.
|
|
||||||
*/
|
|
||||||
public function before($findName, callable $middleware, $withName = '')
|
|
||||||
{
|
|
||||||
$this->splice($findName, $withName, $middleware, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add a middleware after another middleware by name.
|
|
||||||
*
|
|
||||||
* @param string $findName Middleware to find
|
|
||||||
* @param callable $middleware Middleware function
|
|
||||||
* @param string $withName Name to register for this middleware.
|
|
||||||
*/
|
|
||||||
public function after($findName, callable $middleware, $withName = '')
|
|
||||||
{
|
|
||||||
$this->splice($findName, $withName, $middleware, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Remove a middleware by instance or name from the stack.
|
|
||||||
*
|
|
||||||
* @param callable|string $remove Middleware to remove by instance or name.
|
|
||||||
*/
|
|
||||||
public function remove($remove)
|
|
||||||
{
|
|
||||||
$this->cached = null;
|
|
||||||
$idx = is_callable($remove) ? 0 : 1;
|
|
||||||
$this->stack = array_values(array_filter(
|
|
||||||
$this->stack,
|
|
||||||
function ($tuple) use ($idx, $remove) {
|
|
||||||
return $tuple[$idx] !== $remove;
|
|
||||||
}
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Compose the middleware and handler into a single callable function.
|
|
||||||
*
|
|
||||||
* @return callable
|
|
||||||
*/
|
|
||||||
public function resolve()
|
|
||||||
{
|
|
||||||
if (!$this->cached) {
|
|
||||||
if (!($prev = $this->handler)) {
|
|
||||||
throw new \LogicException('No handler has been specified');
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (array_reverse($this->stack) as $fn) {
|
|
||||||
$prev = $fn[0]($prev);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->cached = $prev;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $this->cached;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $name
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
private function findByName($name)
|
|
||||||
{
|
|
||||||
foreach ($this->stack as $k => $v) {
|
|
||||||
if ($v[1] === $name) {
|
|
||||||
return $k;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new \InvalidArgumentException("Middleware not found: $name");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Splices a function into the middleware list at a specific position.
|
|
||||||
*
|
|
||||||
* @param string $findName
|
|
||||||
* @param string $withName
|
|
||||||
* @param callable $middleware
|
|
||||||
* @param bool $before
|
|
||||||
*/
|
|
||||||
private function splice($findName, $withName, callable $middleware, $before)
|
|
||||||
{
|
|
||||||
$this->cached = null;
|
|
||||||
$idx = $this->findByName($findName);
|
|
||||||
$tuple = [$middleware, $withName];
|
|
||||||
|
|
||||||
if ($before) {
|
|
||||||
if ($idx === 0) {
|
|
||||||
array_unshift($this->stack, $tuple);
|
|
||||||
} else {
|
|
||||||
$replacement = [$tuple, $this->stack[$idx]];
|
|
||||||
array_splice($this->stack, $idx, 1, $replacement);
|
|
||||||
}
|
|
||||||
} elseif ($idx === count($this->stack) - 1) {
|
|
||||||
$this->stack[] = $tuple;
|
|
||||||
} else {
|
|
||||||
$replacement = [$this->stack[$idx], $tuple];
|
|
||||||
array_splice($this->stack, $idx, 1, $replacement);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Provides a debug string for a given callable.
|
|
||||||
*
|
|
||||||
* @param array|callable $fn Function to write as a string.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
private function debugCallable($fn)
|
|
||||||
{
|
|
||||||
if (is_string($fn)) {
|
|
||||||
return "callable({$fn})";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (is_array($fn)) {
|
|
||||||
return is_string($fn[0])
|
|
||||||
? "callable({$fn[0]}::{$fn[1]})"
|
|
||||||
: "callable(['" . get_class($fn[0]) . "', '{$fn[1]}'])";
|
|
||||||
}
|
|
||||||
|
|
||||||
return 'callable(' . spl_object_hash($fn) . ')';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-185
@@ -1,185 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use Psr\Http\Message\MessageInterface;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Formats log messages using variable substitutions for requests, responses,
|
|
||||||
* and other transactional data.
|
|
||||||
*
|
|
||||||
* The following variable substitutions are supported:
|
|
||||||
*
|
|
||||||
* - {request}: Full HTTP request message
|
|
||||||
* - {response}: Full HTTP response message
|
|
||||||
* - {ts}: ISO 8601 date in GMT
|
|
||||||
* - {date_iso_8601} ISO 8601 date in GMT
|
|
||||||
* - {date_common_log} Apache common log date using the configured timezone.
|
|
||||||
* - {host}: Host of the request
|
|
||||||
* - {method}: Method of the request
|
|
||||||
* - {uri}: URI of the request
|
|
||||||
* - {version}: Protocol version
|
|
||||||
* - {target}: Request target of the request (path + query + fragment)
|
|
||||||
* - {hostname}: Hostname of the machine that sent the request
|
|
||||||
* - {code}: Status code of the response (if available)
|
|
||||||
* - {phrase}: Reason phrase of the response (if available)
|
|
||||||
* - {error}: Any error messages (if available)
|
|
||||||
* - {req_header_*}: Replace `*` with the lowercased name of a request header to add to the message
|
|
||||||
* - {res_header_*}: Replace `*` with the lowercased name of a response header to add to the message
|
|
||||||
* - {req_headers}: Request headers
|
|
||||||
* - {res_headers}: Response headers
|
|
||||||
* - {req_body}: Request body
|
|
||||||
* - {res_body}: Response body
|
|
||||||
*/
|
|
||||||
class MessageFormatter
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Apache Common Log Format.
|
|
||||||
* @link http://httpd.apache.org/docs/2.4/logs.html#common
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
const CLF = "{hostname} {req_header_User-Agent} - [{date_common_log}] \"{method} {target} HTTP/{version}\" {code} {res_header_Content-Length}";
|
|
||||||
const DEBUG = ">>>>>>>>\n{request}\n<<<<<<<<\n{response}\n--------\n{error}";
|
|
||||||
const SHORT = '[{ts}] "{method} {target} HTTP/{version}" {code}';
|
|
||||||
|
|
||||||
/** @var string Template used to format log messages */
|
|
||||||
private $template;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $template Log message template
|
|
||||||
*/
|
|
||||||
public function __construct($template = self::CLF)
|
|
||||||
{
|
|
||||||
$this->template = $template ?: self::CLF;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns a formatted message string.
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request Request that was sent
|
|
||||||
* @param ResponseInterface $response Response that was received
|
|
||||||
* @param \Exception $error Exception that was received
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function format(
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response = null,
|
|
||||||
\Exception $error = null
|
|
||||||
) {
|
|
||||||
$cache = [];
|
|
||||||
|
|
||||||
return preg_replace_callback(
|
|
||||||
'/{\s*([A-Za-z_\-\.0-9]+)\s*}/',
|
|
||||||
function (array $matches) use ($request, $response, $error, &$cache) {
|
|
||||||
if (isset($cache[$matches[1]])) {
|
|
||||||
return $cache[$matches[1]];
|
|
||||||
}
|
|
||||||
|
|
||||||
$result = '';
|
|
||||||
switch ($matches[1]) {
|
|
||||||
case 'request':
|
|
||||||
$result = Psr7\str($request);
|
|
||||||
break;
|
|
||||||
case 'response':
|
|
||||||
$result = $response ? Psr7\str($response) : '';
|
|
||||||
break;
|
|
||||||
case 'req_headers':
|
|
||||||
$result = trim($request->getMethod()
|
|
||||||
. ' ' . $request->getRequestTarget())
|
|
||||||
. ' HTTP/' . $request->getProtocolVersion() . "\r\n"
|
|
||||||
. $this->headers($request);
|
|
||||||
break;
|
|
||||||
case 'res_headers':
|
|
||||||
$result = $response ?
|
|
||||||
sprintf(
|
|
||||||
'HTTP/%s %d %s',
|
|
||||||
$response->getProtocolVersion(),
|
|
||||||
$response->getStatusCode(),
|
|
||||||
$response->getReasonPhrase()
|
|
||||||
) . "\r\n" . $this->headers($response)
|
|
||||||
: 'NULL';
|
|
||||||
break;
|
|
||||||
case 'req_body':
|
|
||||||
$result = $request->getBody();
|
|
||||||
break;
|
|
||||||
case 'res_body':
|
|
||||||
$result = $response ? $response->getBody() : 'NULL';
|
|
||||||
break;
|
|
||||||
case 'ts':
|
|
||||||
case 'date_iso_8601':
|
|
||||||
$result = gmdate('c');
|
|
||||||
break;
|
|
||||||
case 'date_common_log':
|
|
||||||
$result = date('d/M/Y:H:i:s O');
|
|
||||||
break;
|
|
||||||
case 'method':
|
|
||||||
$result = $request->getMethod();
|
|
||||||
break;
|
|
||||||
case 'version':
|
|
||||||
$result = $request->getProtocolVersion();
|
|
||||||
break;
|
|
||||||
case 'uri':
|
|
||||||
case 'url':
|
|
||||||
$result = $request->getUri();
|
|
||||||
break;
|
|
||||||
case 'target':
|
|
||||||
$result = $request->getRequestTarget();
|
|
||||||
break;
|
|
||||||
case 'req_version':
|
|
||||||
$result = $request->getProtocolVersion();
|
|
||||||
break;
|
|
||||||
case 'res_version':
|
|
||||||
$result = $response
|
|
||||||
? $response->getProtocolVersion()
|
|
||||||
: 'NULL';
|
|
||||||
break;
|
|
||||||
case 'host':
|
|
||||||
$result = $request->getHeaderLine('Host');
|
|
||||||
break;
|
|
||||||
case 'hostname':
|
|
||||||
$result = gethostname();
|
|
||||||
break;
|
|
||||||
case 'code':
|
|
||||||
$result = $response ? $response->getStatusCode() : 'NULL';
|
|
||||||
break;
|
|
||||||
case 'phrase':
|
|
||||||
$result = $response ? $response->getReasonPhrase() : 'NULL';
|
|
||||||
break;
|
|
||||||
case 'error':
|
|
||||||
$result = $error ? $error->getMessage() : 'NULL';
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
// handle prefixed dynamic headers
|
|
||||||
if (strpos($matches[1], 'req_header_') === 0) {
|
|
||||||
$result = $request->getHeaderLine(substr($matches[1], 11));
|
|
||||||
} elseif (strpos($matches[1], 'res_header_') === 0) {
|
|
||||||
$result = $response
|
|
||||||
? $response->getHeaderLine(substr($matches[1], 11))
|
|
||||||
: 'NULL';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$cache[$matches[1]] = $result;
|
|
||||||
return $result;
|
|
||||||
},
|
|
||||||
$this->template
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get headers from message as string
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
private function headers(MessageInterface $message)
|
|
||||||
{
|
|
||||||
$result = '';
|
|
||||||
foreach ($message->getHeaders() as $name => $values) {
|
|
||||||
$result .= $name . ': ' . implode(', ', $values) . "\r\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
return trim($result);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-254
@@ -1,254 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Cookie\CookieJarInterface;
|
|
||||||
use GuzzleHttp\Exception\RequestException;
|
|
||||||
use GuzzleHttp\Promise\RejectedPromise;
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Log\LoggerInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Functions used to create and wrap handlers with handler middleware.
|
|
||||||
*/
|
|
||||||
final class Middleware
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Middleware that adds cookies to requests.
|
|
||||||
*
|
|
||||||
* The options array must be set to a CookieJarInterface in order to use
|
|
||||||
* cookies. This is typically handled for you by a client.
|
|
||||||
*
|
|
||||||
* @return callable Returns a function that accepts the next handler.
|
|
||||||
*/
|
|
||||||
public static function cookies()
|
|
||||||
{
|
|
||||||
return function (callable $handler) {
|
|
||||||
return function ($request, array $options) use ($handler) {
|
|
||||||
if (empty($options['cookies'])) {
|
|
||||||
return $handler($request, $options);
|
|
||||||
} elseif (!($options['cookies'] instanceof CookieJarInterface)) {
|
|
||||||
throw new \InvalidArgumentException('cookies must be an instance of GuzzleHttp\Cookie\CookieJarInterface');
|
|
||||||
}
|
|
||||||
$cookieJar = $options['cookies'];
|
|
||||||
$request = $cookieJar->withCookieHeader($request);
|
|
||||||
return $handler($request, $options)
|
|
||||||
->then(
|
|
||||||
function ($response) use ($cookieJar, $request) {
|
|
||||||
$cookieJar->extractCookies($request, $response);
|
|
||||||
return $response;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that throws exceptions for 4xx or 5xx responses when the
|
|
||||||
* "http_error" request option is set to true.
|
|
||||||
*
|
|
||||||
* @return callable Returns a function that accepts the next handler.
|
|
||||||
*/
|
|
||||||
public static function httpErrors()
|
|
||||||
{
|
|
||||||
return function (callable $handler) {
|
|
||||||
return function ($request, array $options) use ($handler) {
|
|
||||||
if (empty($options['http_errors'])) {
|
|
||||||
return $handler($request, $options);
|
|
||||||
}
|
|
||||||
return $handler($request, $options)->then(
|
|
||||||
function (ResponseInterface $response) use ($request) {
|
|
||||||
$code = $response->getStatusCode();
|
|
||||||
if ($code < 400) {
|
|
||||||
return $response;
|
|
||||||
}
|
|
||||||
throw RequestException::create($request, $response);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that pushes history data to an ArrayAccess container.
|
|
||||||
*
|
|
||||||
* @param array|\ArrayAccess $container Container to hold the history (by reference).
|
|
||||||
*
|
|
||||||
* @return callable Returns a function that accepts the next handler.
|
|
||||||
* @throws \InvalidArgumentException if container is not an array or ArrayAccess.
|
|
||||||
*/
|
|
||||||
public static function history(&$container)
|
|
||||||
{
|
|
||||||
if (!is_array($container) && !$container instanceof \ArrayAccess) {
|
|
||||||
throw new \InvalidArgumentException('history container must be an array or object implementing ArrayAccess');
|
|
||||||
}
|
|
||||||
|
|
||||||
return function (callable $handler) use (&$container) {
|
|
||||||
return function ($request, array $options) use ($handler, &$container) {
|
|
||||||
return $handler($request, $options)->then(
|
|
||||||
function ($value) use ($request, &$container, $options) {
|
|
||||||
$container[] = [
|
|
||||||
'request' => $request,
|
|
||||||
'response' => $value,
|
|
||||||
'error' => null,
|
|
||||||
'options' => $options
|
|
||||||
];
|
|
||||||
return $value;
|
|
||||||
},
|
|
||||||
function ($reason) use ($request, &$container, $options) {
|
|
||||||
$container[] = [
|
|
||||||
'request' => $request,
|
|
||||||
'response' => null,
|
|
||||||
'error' => $reason,
|
|
||||||
'options' => $options
|
|
||||||
];
|
|
||||||
return \GuzzleHttp\Promise\rejection_for($reason);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that invokes a callback before and after sending a request.
|
|
||||||
*
|
|
||||||
* The provided listener cannot modify or alter the response. It simply
|
|
||||||
* "taps" into the chain to be notified before returning the promise. The
|
|
||||||
* before listener accepts a request and options array, and the after
|
|
||||||
* listener accepts a request, options array, and response promise.
|
|
||||||
*
|
|
||||||
* @param callable $before Function to invoke before forwarding the request.
|
|
||||||
* @param callable $after Function invoked after forwarding.
|
|
||||||
*
|
|
||||||
* @return callable Returns a function that accepts the next handler.
|
|
||||||
*/
|
|
||||||
public static function tap(callable $before = null, callable $after = null)
|
|
||||||
{
|
|
||||||
return function (callable $handler) use ($before, $after) {
|
|
||||||
return function ($request, array $options) use ($handler, $before, $after) {
|
|
||||||
if ($before) {
|
|
||||||
$before($request, $options);
|
|
||||||
}
|
|
||||||
$response = $handler($request, $options);
|
|
||||||
if ($after) {
|
|
||||||
$after($request, $options, $response);
|
|
||||||
}
|
|
||||||
return $response;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that handles request redirects.
|
|
||||||
*
|
|
||||||
* @return callable Returns a function that accepts the next handler.
|
|
||||||
*/
|
|
||||||
public static function redirect()
|
|
||||||
{
|
|
||||||
return function (callable $handler) {
|
|
||||||
return new RedirectMiddleware($handler);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that retries requests based on the boolean result of
|
|
||||||
* invoking the provided "decider" function.
|
|
||||||
*
|
|
||||||
* If no delay function is provided, a simple implementation of exponential
|
|
||||||
* backoff will be utilized.
|
|
||||||
*
|
|
||||||
* @param callable $decider Function that accepts the number of retries,
|
|
||||||
* a request, [response], and [exception] and
|
|
||||||
* returns true if the request is to be retried.
|
|
||||||
* @param callable $delay Function that accepts the number of retries and
|
|
||||||
* returns the number of milliseconds to delay.
|
|
||||||
*
|
|
||||||
* @return callable Returns a function that accepts the next handler.
|
|
||||||
*/
|
|
||||||
public static function retry(callable $decider, callable $delay = null)
|
|
||||||
{
|
|
||||||
return function (callable $handler) use ($decider, $delay) {
|
|
||||||
return new RetryMiddleware($decider, $handler, $delay);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that logs requests, responses, and errors using a message
|
|
||||||
* formatter.
|
|
||||||
*
|
|
||||||
* @param LoggerInterface $logger Logs messages.
|
|
||||||
* @param MessageFormatter $formatter Formatter used to create message strings.
|
|
||||||
* @param string $logLevel Level at which to log requests.
|
|
||||||
*
|
|
||||||
* @return callable Returns a function that accepts the next handler.
|
|
||||||
*/
|
|
||||||
public static function log(LoggerInterface $logger, MessageFormatter $formatter, $logLevel = 'info' /* \Psr\Log\LogLevel::INFO */)
|
|
||||||
{
|
|
||||||
return function (callable $handler) use ($logger, $formatter, $logLevel) {
|
|
||||||
return function ($request, array $options) use ($handler, $logger, $formatter, $logLevel) {
|
|
||||||
return $handler($request, $options)->then(
|
|
||||||
function ($response) use ($logger, $request, $formatter, $logLevel) {
|
|
||||||
$message = $formatter->format($request, $response);
|
|
||||||
$logger->log($logLevel, $message);
|
|
||||||
return $response;
|
|
||||||
},
|
|
||||||
function ($reason) use ($logger, $request, $formatter) {
|
|
||||||
$response = $reason instanceof RequestException
|
|
||||||
? $reason->getResponse()
|
|
||||||
: null;
|
|
||||||
$message = $formatter->format($request, $response, $reason);
|
|
||||||
$logger->notice($message);
|
|
||||||
return \GuzzleHttp\Promise\rejection_for($reason);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This middleware adds a default content-type if possible, a default
|
|
||||||
* content-length or transfer-encoding header, and the expect header.
|
|
||||||
*
|
|
||||||
* @return callable
|
|
||||||
*/
|
|
||||||
public static function prepareBody()
|
|
||||||
{
|
|
||||||
return function (callable $handler) {
|
|
||||||
return new PrepareBodyMiddleware($handler);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that applies a map function to the request before passing to
|
|
||||||
* the next handler.
|
|
||||||
*
|
|
||||||
* @param callable $fn Function that accepts a RequestInterface and returns
|
|
||||||
* a RequestInterface.
|
|
||||||
* @return callable
|
|
||||||
*/
|
|
||||||
public static function mapRequest(callable $fn)
|
|
||||||
{
|
|
||||||
return function (callable $handler) use ($fn) {
|
|
||||||
return function ($request, array $options) use ($handler, $fn) {
|
|
||||||
return $handler($fn($request), $options);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that applies a map function to the resolved promise's
|
|
||||||
* response.
|
|
||||||
*
|
|
||||||
* @param callable $fn Function that accepts a ResponseInterface and
|
|
||||||
* returns a ResponseInterface.
|
|
||||||
* @return callable
|
|
||||||
*/
|
|
||||||
public static function mapResponse(callable $fn)
|
|
||||||
{
|
|
||||||
return function (callable $handler) use ($fn) {
|
|
||||||
return function ($request, array $options) use ($handler, $fn) {
|
|
||||||
return $handler($request, $options)->then($fn);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-134
@@ -1,134 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Promise\EachPromise;
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use GuzzleHttp\Promise\PromisorInterface;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sends an iterator of requests concurrently using a capped pool size.
|
|
||||||
*
|
|
||||||
* The pool will read from an iterator until it is cancelled or until the
|
|
||||||
* iterator is consumed. When a request is yielded, the request is sent after
|
|
||||||
* applying the "request_options" request options (if provided in the ctor).
|
|
||||||
*
|
|
||||||
* When a function is yielded by the iterator, the function is provided the
|
|
||||||
* "request_options" array that should be merged on top of any existing
|
|
||||||
* options, and the function MUST then return a wait-able promise.
|
|
||||||
*/
|
|
||||||
class Pool implements PromisorInterface
|
|
||||||
{
|
|
||||||
/** @var EachPromise */
|
|
||||||
private $each;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ClientInterface $client Client used to send the requests.
|
|
||||||
* @param array|\Iterator $requests Requests or functions that return
|
|
||||||
* requests to send concurrently.
|
|
||||||
* @param array $config Associative array of options
|
|
||||||
* - concurrency: (int) Maximum number of requests to send concurrently
|
|
||||||
* - options: Array of request options to apply to each request.
|
|
||||||
* - fulfilled: (callable) Function to invoke when a request completes.
|
|
||||||
* - rejected: (callable) Function to invoke when a request is rejected.
|
|
||||||
*/
|
|
||||||
public function __construct(
|
|
||||||
ClientInterface $client,
|
|
||||||
$requests,
|
|
||||||
array $config = []
|
|
||||||
) {
|
|
||||||
// Backwards compatibility.
|
|
||||||
if (isset($config['pool_size'])) {
|
|
||||||
$config['concurrency'] = $config['pool_size'];
|
|
||||||
} elseif (!isset($config['concurrency'])) {
|
|
||||||
$config['concurrency'] = 25;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($config['options'])) {
|
|
||||||
$opts = $config['options'];
|
|
||||||
unset($config['options']);
|
|
||||||
} else {
|
|
||||||
$opts = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
$iterable = \GuzzleHttp\Promise\iter_for($requests);
|
|
||||||
$requests = function () use ($iterable, $client, $opts) {
|
|
||||||
foreach ($iterable as $key => $rfn) {
|
|
||||||
if ($rfn instanceof RequestInterface) {
|
|
||||||
yield $key => $client->sendAsync($rfn, $opts);
|
|
||||||
} elseif (is_callable($rfn)) {
|
|
||||||
yield $key => $rfn($opts);
|
|
||||||
} else {
|
|
||||||
throw new \InvalidArgumentException('Each value yielded by '
|
|
||||||
. 'the iterator must be a Psr7\Http\Message\RequestInterface '
|
|
||||||
. 'or a callable that returns a promise that fulfills '
|
|
||||||
. 'with a Psr7\Message\Http\ResponseInterface object.');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
$this->each = new EachPromise($requests(), $config);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get promise
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
public function promise()
|
|
||||||
{
|
|
||||||
return $this->each->promise();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sends multiple requests concurrently and returns an array of responses
|
|
||||||
* and exceptions that uses the same ordering as the provided requests.
|
|
||||||
*
|
|
||||||
* IMPORTANT: This method keeps every request and response in memory, and
|
|
||||||
* as such, is NOT recommended when sending a large number or an
|
|
||||||
* indeterminate number of requests concurrently.
|
|
||||||
*
|
|
||||||
* @param ClientInterface $client Client used to send the requests
|
|
||||||
* @param array|\Iterator $requests Requests to send concurrently.
|
|
||||||
* @param array $options Passes through the options available in
|
|
||||||
* {@see GuzzleHttp\Pool::__construct}
|
|
||||||
*
|
|
||||||
* @return array Returns an array containing the response or an exception
|
|
||||||
* in the same order that the requests were sent.
|
|
||||||
* @throws \InvalidArgumentException if the event format is incorrect.
|
|
||||||
*/
|
|
||||||
public static function batch(
|
|
||||||
ClientInterface $client,
|
|
||||||
$requests,
|
|
||||||
array $options = []
|
|
||||||
) {
|
|
||||||
$res = [];
|
|
||||||
self::cmpCallback($options, 'fulfilled', $res);
|
|
||||||
self::cmpCallback($options, 'rejected', $res);
|
|
||||||
$pool = new static($client, $requests, $options);
|
|
||||||
$pool->promise()->wait();
|
|
||||||
ksort($res);
|
|
||||||
|
|
||||||
return $res;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Execute callback(s)
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
private static function cmpCallback(array &$options, $name, array &$results)
|
|
||||||
{
|
|
||||||
if (!isset($options[$name])) {
|
|
||||||
$options[$name] = function ($v, $k) use (&$results) {
|
|
||||||
$results[$k] = $v;
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
$currentFn = $options[$name];
|
|
||||||
$options[$name] = function ($v, $k) use (&$results, $currentFn) {
|
|
||||||
$currentFn($v, $k);
|
|
||||||
$results[$k] = $v;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,111 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Prepares requests that contain a body, adding the Content-Length,
|
|
||||||
* Content-Type, and Expect headers.
|
|
||||||
*/
|
|
||||||
class PrepareBodyMiddleware
|
|
||||||
{
|
|
||||||
/** @var callable */
|
|
||||||
private $nextHandler;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param callable $nextHandler Next handler to invoke.
|
|
||||||
*/
|
|
||||||
public function __construct(callable $nextHandler)
|
|
||||||
{
|
|
||||||
$this->nextHandler = $nextHandler;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param array $options
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
$fn = $this->nextHandler;
|
|
||||||
|
|
||||||
// Don't do anything if the request has no body.
|
|
||||||
if ($request->getBody()->getSize() === 0) {
|
|
||||||
return $fn($request, $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
$modify = [];
|
|
||||||
|
|
||||||
// Add a default content-type if possible.
|
|
||||||
if (!$request->hasHeader('Content-Type')) {
|
|
||||||
if ($uri = $request->getBody()->getMetadata('uri')) {
|
|
||||||
if ($type = Psr7\mimetype_from_filename($uri)) {
|
|
||||||
$modify['set_headers']['Content-Type'] = $type;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add a default content-length or transfer-encoding header.
|
|
||||||
if (!$request->hasHeader('Content-Length')
|
|
||||||
&& !$request->hasHeader('Transfer-Encoding')
|
|
||||||
) {
|
|
||||||
$size = $request->getBody()->getSize();
|
|
||||||
if ($size !== null) {
|
|
||||||
$modify['set_headers']['Content-Length'] = $size;
|
|
||||||
} else {
|
|
||||||
$modify['set_headers']['Transfer-Encoding'] = 'chunked';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add the expect header if needed.
|
|
||||||
$this->addExpectHeader($request, $options, $modify);
|
|
||||||
|
|
||||||
return $fn(Psr7\modify_request($request, $modify), $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add expect header
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
private function addExpectHeader(
|
|
||||||
RequestInterface $request,
|
|
||||||
array $options,
|
|
||||||
array &$modify
|
|
||||||
) {
|
|
||||||
// Determine if the Expect header should be used
|
|
||||||
if ($request->hasHeader('Expect')) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$expect = isset($options['expect']) ? $options['expect'] : null;
|
|
||||||
|
|
||||||
// Return if disabled or if you're not using HTTP/1.1 or HTTP/2.0
|
|
||||||
if ($expect === false || $request->getProtocolVersion() < 1.1) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// The expect header is unconditionally enabled
|
|
||||||
if ($expect === true) {
|
|
||||||
$modify['set_headers']['Expect'] = '100-Continue';
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// By default, send the expect header when the payload is > 1mb
|
|
||||||
if ($expect === null) {
|
|
||||||
$expect = 1048576;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Always add if the body cannot be rewound, the size cannot be
|
|
||||||
// determined, or the size is greater than the cutoff threshold
|
|
||||||
$body = $request->getBody();
|
|
||||||
$size = $body->getSize();
|
|
||||||
|
|
||||||
if ($size === null || $size >= (int) $expect || !$body->isSeekable()) {
|
|
||||||
$modify['set_headers']['Expect'] = '100-Continue';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-255
@@ -1,255 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Exception\BadResponseException;
|
|
||||||
use GuzzleHttp\Exception\TooManyRedirectsException;
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Http\Message\UriInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Request redirect middleware.
|
|
||||||
*
|
|
||||||
* Apply this middleware like other middleware using
|
|
||||||
* {@see \GuzzleHttp\Middleware::redirect()}.
|
|
||||||
*/
|
|
||||||
class RedirectMiddleware
|
|
||||||
{
|
|
||||||
const HISTORY_HEADER = 'X-Guzzle-Redirect-History';
|
|
||||||
|
|
||||||
const STATUS_HISTORY_HEADER = 'X-Guzzle-Redirect-Status-History';
|
|
||||||
|
|
||||||
public static $defaultSettings = [
|
|
||||||
'max' => 5,
|
|
||||||
'protocols' => ['http', 'https'],
|
|
||||||
'strict' => false,
|
|
||||||
'referer' => false,
|
|
||||||
'track_redirects' => false,
|
|
||||||
];
|
|
||||||
|
|
||||||
/** @var callable */
|
|
||||||
private $nextHandler;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param callable $nextHandler Next handler to invoke.
|
|
||||||
*/
|
|
||||||
public function __construct(callable $nextHandler)
|
|
||||||
{
|
|
||||||
$this->nextHandler = $nextHandler;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param array $options
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
$fn = $this->nextHandler;
|
|
||||||
|
|
||||||
if (empty($options['allow_redirects'])) {
|
|
||||||
return $fn($request, $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($options['allow_redirects'] === true) {
|
|
||||||
$options['allow_redirects'] = self::$defaultSettings;
|
|
||||||
} elseif (!is_array($options['allow_redirects'])) {
|
|
||||||
throw new \InvalidArgumentException('allow_redirects must be true, false, or array');
|
|
||||||
} else {
|
|
||||||
// Merge the default settings with the provided settings
|
|
||||||
$options['allow_redirects'] += self::$defaultSettings;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (empty($options['allow_redirects']['max'])) {
|
|
||||||
return $fn($request, $options);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $fn($request, $options)
|
|
||||||
->then(function (ResponseInterface $response) use ($request, $options) {
|
|
||||||
return $this->checkRedirect($request, $options, $response);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param array $options
|
|
||||||
* @param ResponseInterface $response
|
|
||||||
*
|
|
||||||
* @return ResponseInterface|PromiseInterface
|
|
||||||
*/
|
|
||||||
public function checkRedirect(
|
|
||||||
RequestInterface $request,
|
|
||||||
array $options,
|
|
||||||
ResponseInterface $response
|
|
||||||
) {
|
|
||||||
if (substr($response->getStatusCode(), 0, 1) != '3'
|
|
||||||
|| !$response->hasHeader('Location')
|
|
||||||
) {
|
|
||||||
return $response;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->guardMax($request, $options);
|
|
||||||
$nextRequest = $this->modifyRequest($request, $options, $response);
|
|
||||||
|
|
||||||
if (isset($options['allow_redirects']['on_redirect'])) {
|
|
||||||
call_user_func(
|
|
||||||
$options['allow_redirects']['on_redirect'],
|
|
||||||
$request,
|
|
||||||
$response,
|
|
||||||
$nextRequest->getUri()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** @var PromiseInterface|ResponseInterface $promise */
|
|
||||||
$promise = $this($nextRequest, $options);
|
|
||||||
|
|
||||||
// Add headers to be able to track history of redirects.
|
|
||||||
if (!empty($options['allow_redirects']['track_redirects'])) {
|
|
||||||
return $this->withTracking(
|
|
||||||
$promise,
|
|
||||||
(string) $nextRequest->getUri(),
|
|
||||||
$response->getStatusCode()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $promise;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Enable tracking on promise.
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
private function withTracking(PromiseInterface $promise, $uri, $statusCode)
|
|
||||||
{
|
|
||||||
return $promise->then(
|
|
||||||
function (ResponseInterface $response) use ($uri, $statusCode) {
|
|
||||||
// Note that we are pushing to the front of the list as this
|
|
||||||
// would be an earlier response than what is currently present
|
|
||||||
// in the history header.
|
|
||||||
$historyHeader = $response->getHeader(self::HISTORY_HEADER);
|
|
||||||
$statusHeader = $response->getHeader(self::STATUS_HISTORY_HEADER);
|
|
||||||
array_unshift($historyHeader, $uri);
|
|
||||||
array_unshift($statusHeader, $statusCode);
|
|
||||||
return $response->withHeader(self::HISTORY_HEADER, $historyHeader)
|
|
||||||
->withHeader(self::STATUS_HISTORY_HEADER, $statusHeader);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Check for too many redirects
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*
|
|
||||||
* @throws TooManyRedirectsException Too many redirects.
|
|
||||||
*/
|
|
||||||
private function guardMax(RequestInterface $request, array &$options)
|
|
||||||
{
|
|
||||||
$current = isset($options['__redirect_count'])
|
|
||||||
? $options['__redirect_count']
|
|
||||||
: 0;
|
|
||||||
$options['__redirect_count'] = $current + 1;
|
|
||||||
$max = $options['allow_redirects']['max'];
|
|
||||||
|
|
||||||
if ($options['__redirect_count'] > $max) {
|
|
||||||
throw new TooManyRedirectsException(
|
|
||||||
"Will not follow more than {$max} redirects",
|
|
||||||
$request
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param array $options
|
|
||||||
* @param ResponseInterface $response
|
|
||||||
*
|
|
||||||
* @return RequestInterface
|
|
||||||
*/
|
|
||||||
public function modifyRequest(
|
|
||||||
RequestInterface $request,
|
|
||||||
array $options,
|
|
||||||
ResponseInterface $response
|
|
||||||
) {
|
|
||||||
// Request modifications to apply.
|
|
||||||
$modify = [];
|
|
||||||
$protocols = $options['allow_redirects']['protocols'];
|
|
||||||
|
|
||||||
// Use a GET request if this is an entity enclosing request and we are
|
|
||||||
// not forcing RFC compliance, but rather emulating what all browsers
|
|
||||||
// would do.
|
|
||||||
$statusCode = $response->getStatusCode();
|
|
||||||
if ($statusCode == 303 ||
|
|
||||||
($statusCode <= 302 && !$options['allow_redirects']['strict'])
|
|
||||||
) {
|
|
||||||
$modify['method'] = 'GET';
|
|
||||||
$modify['body'] = '';
|
|
||||||
}
|
|
||||||
|
|
||||||
$uri = $this->redirectUri($request, $response, $protocols);
|
|
||||||
if (isset($options['idn_conversion']) && ($options['idn_conversion'] !== false)) {
|
|
||||||
$idnOptions = ($options['idn_conversion'] === true) ? IDNA_DEFAULT : $options['idn_conversion'];
|
|
||||||
$uri = Utils::idnUriConvert($uri, $idnOptions);
|
|
||||||
}
|
|
||||||
|
|
||||||
$modify['uri'] = $uri;
|
|
||||||
Psr7\rewind_body($request);
|
|
||||||
|
|
||||||
// Add the Referer header if it is told to do so and only
|
|
||||||
// add the header if we are not redirecting from https to http.
|
|
||||||
if ($options['allow_redirects']['referer']
|
|
||||||
&& $modify['uri']->getScheme() === $request->getUri()->getScheme()
|
|
||||||
) {
|
|
||||||
$uri = $request->getUri()->withUserInfo('');
|
|
||||||
$modify['set_headers']['Referer'] = (string) $uri;
|
|
||||||
} else {
|
|
||||||
$modify['remove_headers'][] = 'Referer';
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove Authorization header if host is different.
|
|
||||||
if ($request->getUri()->getHost() !== $modify['uri']->getHost()) {
|
|
||||||
$modify['remove_headers'][] = 'Authorization';
|
|
||||||
}
|
|
||||||
|
|
||||||
return Psr7\modify_request($request, $modify);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the appropriate URL on the request based on the location header
|
|
||||||
*
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param ResponseInterface $response
|
|
||||||
* @param array $protocols
|
|
||||||
*
|
|
||||||
* @return UriInterface
|
|
||||||
*/
|
|
||||||
private function redirectUri(
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response,
|
|
||||||
array $protocols
|
|
||||||
) {
|
|
||||||
$location = Psr7\UriResolver::resolve(
|
|
||||||
$request->getUri(),
|
|
||||||
new Psr7\Uri($response->getHeaderLine('Location'))
|
|
||||||
);
|
|
||||||
|
|
||||||
// Ensure that the redirect URI is allowed based on the protocols.
|
|
||||||
if (!in_array($location->getScheme(), $protocols)) {
|
|
||||||
throw new BadResponseException(
|
|
||||||
sprintf(
|
|
||||||
'Redirect URI, %s, does not use one of the allowed redirect protocols: %s',
|
|
||||||
$location,
|
|
||||||
implode(', ', $protocols)
|
|
||||||
),
|
|
||||||
$request,
|
|
||||||
$response
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $location;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-263
@@ -1,263 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This class contains a list of built-in Guzzle request options.
|
|
||||||
*
|
|
||||||
* More documentation for each option can be found at http://guzzlephp.org/.
|
|
||||||
*
|
|
||||||
* @link http://docs.guzzlephp.org/en/v6/request-options.html
|
|
||||||
*/
|
|
||||||
final class RequestOptions
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* allow_redirects: (bool|array) Controls redirect behavior. Pass false
|
|
||||||
* to disable redirects, pass true to enable redirects, pass an
|
|
||||||
* associative to provide custom redirect settings. Defaults to "false".
|
|
||||||
* This option only works if your handler has the RedirectMiddleware. When
|
|
||||||
* passing an associative array, you can provide the following key value
|
|
||||||
* pairs:
|
|
||||||
*
|
|
||||||
* - max: (int, default=5) maximum number of allowed redirects.
|
|
||||||
* - strict: (bool, default=false) Set to true to use strict redirects
|
|
||||||
* meaning redirect POST requests with POST requests vs. doing what most
|
|
||||||
* browsers do which is redirect POST requests with GET requests
|
|
||||||
* - referer: (bool, default=false) Set to true to enable the Referer
|
|
||||||
* header.
|
|
||||||
* - protocols: (array, default=['http', 'https']) Allowed redirect
|
|
||||||
* protocols.
|
|
||||||
* - on_redirect: (callable) PHP callable that is invoked when a redirect
|
|
||||||
* is encountered. The callable is invoked with the request, the redirect
|
|
||||||
* response that was received, and the effective URI. Any return value
|
|
||||||
* from the on_redirect function is ignored.
|
|
||||||
*/
|
|
||||||
const ALLOW_REDIRECTS = 'allow_redirects';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* auth: (array) Pass an array of HTTP authentication parameters to use
|
|
||||||
* with the request. The array must contain the username in index [0],
|
|
||||||
* the password in index [1], and you can optionally provide a built-in
|
|
||||||
* authentication type in index [2]. Pass null to disable authentication
|
|
||||||
* for a request.
|
|
||||||
*/
|
|
||||||
const AUTH = 'auth';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* body: (resource|string|null|int|float|StreamInterface|callable|\Iterator)
|
|
||||||
* Body to send in the request.
|
|
||||||
*/
|
|
||||||
const BODY = 'body';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* cert: (string|array) Set to a string to specify the path to a file
|
|
||||||
* containing a PEM formatted SSL client side certificate. If a password
|
|
||||||
* is required, then set cert to an array containing the path to the PEM
|
|
||||||
* file in the first array element followed by the certificate password
|
|
||||||
* in the second array element.
|
|
||||||
*/
|
|
||||||
const CERT = 'cert';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* cookies: (bool|GuzzleHttp\Cookie\CookieJarInterface, default=false)
|
|
||||||
* Specifies whether or not cookies are used in a request or what cookie
|
|
||||||
* jar to use or what cookies to send. This option only works if your
|
|
||||||
* handler has the `cookie` middleware. Valid values are `false` and
|
|
||||||
* an instance of {@see GuzzleHttp\Cookie\CookieJarInterface}.
|
|
||||||
*/
|
|
||||||
const COOKIES = 'cookies';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* connect_timeout: (float, default=0) Float describing the number of
|
|
||||||
* seconds to wait while trying to connect to a server. Use 0 to wait
|
|
||||||
* indefinitely (the default behavior).
|
|
||||||
*/
|
|
||||||
const CONNECT_TIMEOUT = 'connect_timeout';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* debug: (bool|resource) Set to true or set to a PHP stream returned by
|
|
||||||
* fopen() enable debug output with the HTTP handler used to send a
|
|
||||||
* request.
|
|
||||||
*/
|
|
||||||
const DEBUG = 'debug';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* decode_content: (bool, default=true) Specify whether or not
|
|
||||||
* Content-Encoding responses (gzip, deflate, etc.) are automatically
|
|
||||||
* decoded.
|
|
||||||
*/
|
|
||||||
const DECODE_CONTENT = 'decode_content';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* delay: (int) The amount of time to delay before sending in milliseconds.
|
|
||||||
*/
|
|
||||||
const DELAY = 'delay';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* expect: (bool|integer) Controls the behavior of the
|
|
||||||
* "Expect: 100-Continue" header.
|
|
||||||
*
|
|
||||||
* Set to `true` to enable the "Expect: 100-Continue" header for all
|
|
||||||
* requests that sends a body. Set to `false` to disable the
|
|
||||||
* "Expect: 100-Continue" header for all requests. Set to a number so that
|
|
||||||
* the size of the payload must be greater than the number in order to send
|
|
||||||
* the Expect header. Setting to a number will send the Expect header for
|
|
||||||
* all requests in which the size of the payload cannot be determined or
|
|
||||||
* where the body is not rewindable.
|
|
||||||
*
|
|
||||||
* By default, Guzzle will add the "Expect: 100-Continue" header when the
|
|
||||||
* size of the body of a request is greater than 1 MB and a request is
|
|
||||||
* using HTTP/1.1.
|
|
||||||
*/
|
|
||||||
const EXPECT = 'expect';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* form_params: (array) Associative array of form field names to values
|
|
||||||
* where each value is a string or array of strings. Sets the Content-Type
|
|
||||||
* header to application/x-www-form-urlencoded when no Content-Type header
|
|
||||||
* is already present.
|
|
||||||
*/
|
|
||||||
const FORM_PARAMS = 'form_params';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* headers: (array) Associative array of HTTP headers. Each value MUST be
|
|
||||||
* a string or array of strings.
|
|
||||||
*/
|
|
||||||
const HEADERS = 'headers';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* http_errors: (bool, default=true) Set to false to disable exceptions
|
|
||||||
* when a non- successful HTTP response is received. By default,
|
|
||||||
* exceptions will be thrown for 4xx and 5xx responses. This option only
|
|
||||||
* works if your handler has the `httpErrors` middleware.
|
|
||||||
*/
|
|
||||||
const HTTP_ERRORS = 'http_errors';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* idn: (bool|int, default=true) A combination of IDNA_* constants for
|
|
||||||
* idn_to_ascii() PHP's function (see "options" parameter). Set to false to
|
|
||||||
* disable IDN support completely, or to true to use the default
|
|
||||||
* configuration (IDNA_DEFAULT constant).
|
|
||||||
*/
|
|
||||||
const IDN_CONVERSION = 'idn_conversion';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* json: (mixed) Adds JSON data to a request. The provided value is JSON
|
|
||||||
* encoded and a Content-Type header of application/json will be added to
|
|
||||||
* the request if no Content-Type header is already present.
|
|
||||||
*/
|
|
||||||
const JSON = 'json';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* multipart: (array) Array of associative arrays, each containing a
|
|
||||||
* required "name" key mapping to the form field, name, a required
|
|
||||||
* "contents" key mapping to a StreamInterface|resource|string, an
|
|
||||||
* optional "headers" associative array of custom headers, and an
|
|
||||||
* optional "filename" key mapping to a string to send as the filename in
|
|
||||||
* the part. If no "filename" key is present, then no "filename" attribute
|
|
||||||
* will be added to the part.
|
|
||||||
*/
|
|
||||||
const MULTIPART = 'multipart';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* on_headers: (callable) A callable that is invoked when the HTTP headers
|
|
||||||
* of the response have been received but the body has not yet begun to
|
|
||||||
* download.
|
|
||||||
*/
|
|
||||||
const ON_HEADERS = 'on_headers';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* on_stats: (callable) allows you to get access to transfer statistics of
|
|
||||||
* a request and access the lower level transfer details of the handler
|
|
||||||
* associated with your client. ``on_stats`` is a callable that is invoked
|
|
||||||
* when a handler has finished sending a request. The callback is invoked
|
|
||||||
* with transfer statistics about the request, the response received, or
|
|
||||||
* the error encountered. Included in the data is the total amount of time
|
|
||||||
* taken to send the request.
|
|
||||||
*/
|
|
||||||
const ON_STATS = 'on_stats';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* progress: (callable) Defines a function to invoke when transfer
|
|
||||||
* progress is made. The function accepts the following positional
|
|
||||||
* arguments: the total number of bytes expected to be downloaded, the
|
|
||||||
* number of bytes downloaded so far, the number of bytes expected to be
|
|
||||||
* uploaded, the number of bytes uploaded so far.
|
|
||||||
*/
|
|
||||||
const PROGRESS = 'progress';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* proxy: (string|array) Pass a string to specify an HTTP proxy, or an
|
|
||||||
* array to specify different proxies for different protocols (where the
|
|
||||||
* key is the protocol and the value is a proxy string).
|
|
||||||
*/
|
|
||||||
const PROXY = 'proxy';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* query: (array|string) Associative array of query string values to add
|
|
||||||
* to the request. This option uses PHP's http_build_query() to create
|
|
||||||
* the string representation. Pass a string value if you need more
|
|
||||||
* control than what this method provides
|
|
||||||
*/
|
|
||||||
const QUERY = 'query';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* sink: (resource|string|StreamInterface) Where the data of the
|
|
||||||
* response is written to. Defaults to a PHP temp stream. Providing a
|
|
||||||
* string will write data to a file by the given name.
|
|
||||||
*/
|
|
||||||
const SINK = 'sink';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* synchronous: (bool) Set to true to inform HTTP handlers that you intend
|
|
||||||
* on waiting on the response. This can be useful for optimizations. Note
|
|
||||||
* that a promise is still returned if you are using one of the async
|
|
||||||
* client methods.
|
|
||||||
*/
|
|
||||||
const SYNCHRONOUS = 'synchronous';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* ssl_key: (array|string) Specify the path to a file containing a private
|
|
||||||
* SSL key in PEM format. If a password is required, then set to an array
|
|
||||||
* containing the path to the SSL key in the first array element followed
|
|
||||||
* by the password required for the certificate in the second element.
|
|
||||||
*/
|
|
||||||
const SSL_KEY = 'ssl_key';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* stream: Set to true to attempt to stream a response rather than
|
|
||||||
* download it all up-front.
|
|
||||||
*/
|
|
||||||
const STREAM = 'stream';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* verify: (bool|string, default=true) Describes the SSL certificate
|
|
||||||
* verification behavior of a request. Set to true to enable SSL
|
|
||||||
* certificate verification using the system CA bundle when available
|
|
||||||
* (the default). Set to false to disable certificate verification (this
|
|
||||||
* is insecure!). Set to a string to provide the path to a CA bundle on
|
|
||||||
* disk to enable verification using a custom certificate.
|
|
||||||
*/
|
|
||||||
const VERIFY = 'verify';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* timeout: (float, default=0) Float describing the timeout of the
|
|
||||||
* request in seconds. Use 0 to wait indefinitely (the default behavior).
|
|
||||||
*/
|
|
||||||
const TIMEOUT = 'timeout';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* read_timeout: (float, default=default_socket_timeout ini setting) Float describing
|
|
||||||
* the body read timeout, for stream requests.
|
|
||||||
*/
|
|
||||||
const READ_TIMEOUT = 'read_timeout';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* version: (float) Specifies the HTTP protocol version to attempt to use.
|
|
||||||
*/
|
|
||||||
const VERSION = 'version';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* force_ip_resolve: (bool) Force client to use only ipv4 or ipv6 protocol
|
|
||||||
*/
|
|
||||||
const FORCE_IP_RESOLVE = 'force_ip_resolve';
|
|
||||||
}
|
|
||||||
-128
@@ -1,128 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Promise\PromiseInterface;
|
|
||||||
use GuzzleHttp\Promise\RejectedPromise;
|
|
||||||
use GuzzleHttp\Psr7;
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Middleware that retries requests based on the boolean result of
|
|
||||||
* invoking the provided "decider" function.
|
|
||||||
*/
|
|
||||||
class RetryMiddleware
|
|
||||||
{
|
|
||||||
/** @var callable */
|
|
||||||
private $nextHandler;
|
|
||||||
|
|
||||||
/** @var callable */
|
|
||||||
private $decider;
|
|
||||||
|
|
||||||
/** @var callable */
|
|
||||||
private $delay;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param callable $decider Function that accepts the number of retries,
|
|
||||||
* a request, [response], and [exception] and
|
|
||||||
* returns true if the request is to be
|
|
||||||
* retried.
|
|
||||||
* @param callable $nextHandler Next handler to invoke.
|
|
||||||
* @param callable $delay Function that accepts the number of retries
|
|
||||||
* and [response] and returns the number of
|
|
||||||
* milliseconds to delay.
|
|
||||||
*/
|
|
||||||
public function __construct(
|
|
||||||
callable $decider,
|
|
||||||
callable $nextHandler,
|
|
||||||
callable $delay = null
|
|
||||||
) {
|
|
||||||
$this->decider = $decider;
|
|
||||||
$this->nextHandler = $nextHandler;
|
|
||||||
$this->delay = $delay ?: __CLASS__ . '::exponentialDelay';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Default exponential backoff delay function.
|
|
||||||
*
|
|
||||||
* @param int $retries
|
|
||||||
*
|
|
||||||
* @return int milliseconds.
|
|
||||||
*/
|
|
||||||
public static function exponentialDelay($retries)
|
|
||||||
{
|
|
||||||
return (int) pow(2, $retries - 1) * 1000;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param RequestInterface $request
|
|
||||||
* @param array $options
|
|
||||||
*
|
|
||||||
* @return PromiseInterface
|
|
||||||
*/
|
|
||||||
public function __invoke(RequestInterface $request, array $options)
|
|
||||||
{
|
|
||||||
if (!isset($options['retries'])) {
|
|
||||||
$options['retries'] = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
$fn = $this->nextHandler;
|
|
||||||
return $fn($request, $options)
|
|
||||||
->then(
|
|
||||||
$this->onFulfilled($request, $options),
|
|
||||||
$this->onRejected($request, $options)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Execute fulfilled closure
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
private function onFulfilled(RequestInterface $req, array $options)
|
|
||||||
{
|
|
||||||
return function ($value) use ($req, $options) {
|
|
||||||
if (!call_user_func(
|
|
||||||
$this->decider,
|
|
||||||
$options['retries'],
|
|
||||||
$req,
|
|
||||||
$value,
|
|
||||||
null
|
|
||||||
)) {
|
|
||||||
return $value;
|
|
||||||
}
|
|
||||||
return $this->doRetry($req, $options, $value);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Execute rejected closure
|
|
||||||
*
|
|
||||||
* @return callable
|
|
||||||
*/
|
|
||||||
private function onRejected(RequestInterface $req, array $options)
|
|
||||||
{
|
|
||||||
return function ($reason) use ($req, $options) {
|
|
||||||
if (!call_user_func(
|
|
||||||
$this->decider,
|
|
||||||
$options['retries'],
|
|
||||||
$req,
|
|
||||||
null,
|
|
||||||
$reason
|
|
||||||
)) {
|
|
||||||
return \GuzzleHttp\Promise\rejection_for($reason);
|
|
||||||
}
|
|
||||||
return $this->doRetry($req, $options);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return self
|
|
||||||
*/
|
|
||||||
private function doRetry(RequestInterface $request, array $options, ResponseInterface $response = null)
|
|
||||||
{
|
|
||||||
$options['delay'] = call_user_func($this->delay, ++$options['retries'], $response);
|
|
||||||
|
|
||||||
return $this($request, $options);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-126
@@ -1,126 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use Psr\Http\Message\RequestInterface;
|
|
||||||
use Psr\Http\Message\ResponseInterface;
|
|
||||||
use Psr\Http\Message\UriInterface;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Represents data at the point after it was transferred either successfully
|
|
||||||
* or after a network error.
|
|
||||||
*/
|
|
||||||
final class TransferStats
|
|
||||||
{
|
|
||||||
private $request;
|
|
||||||
private $response;
|
|
||||||
private $transferTime;
|
|
||||||
private $handlerStats;
|
|
||||||
private $handlerErrorData;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param RequestInterface $request Request that was sent.
|
|
||||||
* @param ResponseInterface|null $response Response received (if any)
|
|
||||||
* @param float|null $transferTime Total handler transfer time.
|
|
||||||
* @param mixed $handlerErrorData Handler error data.
|
|
||||||
* @param array $handlerStats Handler specific stats.
|
|
||||||
*/
|
|
||||||
public function __construct(
|
|
||||||
RequestInterface $request,
|
|
||||||
ResponseInterface $response = null,
|
|
||||||
$transferTime = null,
|
|
||||||
$handlerErrorData = null,
|
|
||||||
$handlerStats = []
|
|
||||||
) {
|
|
||||||
$this->request = $request;
|
|
||||||
$this->response = $response;
|
|
||||||
$this->transferTime = $transferTime;
|
|
||||||
$this->handlerErrorData = $handlerErrorData;
|
|
||||||
$this->handlerStats = $handlerStats;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return RequestInterface
|
|
||||||
*/
|
|
||||||
public function getRequest()
|
|
||||||
{
|
|
||||||
return $this->request;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the response that was received (if any).
|
|
||||||
*
|
|
||||||
* @return ResponseInterface|null
|
|
||||||
*/
|
|
||||||
public function getResponse()
|
|
||||||
{
|
|
||||||
return $this->response;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns true if a response was received.
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function hasResponse()
|
|
||||||
{
|
|
||||||
return $this->response !== null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets handler specific error data.
|
|
||||||
*
|
|
||||||
* This might be an exception, a integer representing an error code, or
|
|
||||||
* anything else. Relying on this value assumes that you know what handler
|
|
||||||
* you are using.
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function getHandlerErrorData()
|
|
||||||
{
|
|
||||||
return $this->handlerErrorData;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the effective URI the request was sent to.
|
|
||||||
*
|
|
||||||
* @return UriInterface
|
|
||||||
*/
|
|
||||||
public function getEffectiveUri()
|
|
||||||
{
|
|
||||||
return $this->request->getUri();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the estimated time the request was being transferred by the handler.
|
|
||||||
*
|
|
||||||
* @return float|null Time in seconds.
|
|
||||||
*/
|
|
||||||
public function getTransferTime()
|
|
||||||
{
|
|
||||||
return $this->transferTime;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets an array of all of the handler specific transfer data.
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public function getHandlerStats()
|
|
||||||
{
|
|
||||||
return $this->handlerStats;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a specific handler statistic from the handler by name.
|
|
||||||
*
|
|
||||||
* @param string $stat Handler specific transfer stat to retrieve.
|
|
||||||
*
|
|
||||||
* @return mixed|null
|
|
||||||
*/
|
|
||||||
public function getHandlerStat($stat)
|
|
||||||
{
|
|
||||||
return isset($this->handlerStats[$stat])
|
|
||||||
? $this->handlerStats[$stat]
|
|
||||||
: null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-237
@@ -1,237 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Expands URI templates. Userland implementation of PECL uri_template.
|
|
||||||
*
|
|
||||||
* @link http://tools.ietf.org/html/rfc6570
|
|
||||||
*/
|
|
||||||
class UriTemplate
|
|
||||||
{
|
|
||||||
/** @var string URI template */
|
|
||||||
private $template;
|
|
||||||
|
|
||||||
/** @var array Variables to use in the template expansion */
|
|
||||||
private $variables;
|
|
||||||
|
|
||||||
/** @var array Hash for quick operator lookups */
|
|
||||||
private static $operatorHash = [
|
|
||||||
'' => ['prefix' => '', 'joiner' => ',', 'query' => false],
|
|
||||||
'+' => ['prefix' => '', 'joiner' => ',', 'query' => false],
|
|
||||||
'#' => ['prefix' => '#', 'joiner' => ',', 'query' => false],
|
|
||||||
'.' => ['prefix' => '.', 'joiner' => '.', 'query' => false],
|
|
||||||
'/' => ['prefix' => '/', 'joiner' => '/', 'query' => false],
|
|
||||||
';' => ['prefix' => ';', 'joiner' => ';', 'query' => true],
|
|
||||||
'?' => ['prefix' => '?', 'joiner' => '&', 'query' => true],
|
|
||||||
'&' => ['prefix' => '&', 'joiner' => '&', 'query' => true]
|
|
||||||
];
|
|
||||||
|
|
||||||
/** @var array Delimiters */
|
|
||||||
private static $delims = [':', '/', '?', '#', '[', ']', '@', '!', '$',
|
|
||||||
'&', '\'', '(', ')', '*', '+', ',', ';', '='];
|
|
||||||
|
|
||||||
/** @var array Percent encoded delimiters */
|
|
||||||
private static $delimsPct = ['%3A', '%2F', '%3F', '%23', '%5B', '%5D',
|
|
||||||
'%40', '%21', '%24', '%26', '%27', '%28', '%29', '%2A', '%2B', '%2C',
|
|
||||||
'%3B', '%3D'];
|
|
||||||
|
|
||||||
public function expand($template, array $variables)
|
|
||||||
{
|
|
||||||
if (false === strpos($template, '{')) {
|
|
||||||
return $template;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->template = $template;
|
|
||||||
$this->variables = $variables;
|
|
||||||
|
|
||||||
return preg_replace_callback(
|
|
||||||
'/\{([^\}]+)\}/',
|
|
||||||
[$this, 'expandMatch'],
|
|
||||||
$this->template
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Parse an expression into parts
|
|
||||||
*
|
|
||||||
* @param string $expression Expression to parse
|
|
||||||
*
|
|
||||||
* @return array Returns an associative array of parts
|
|
||||||
*/
|
|
||||||
private function parseExpression($expression)
|
|
||||||
{
|
|
||||||
$result = [];
|
|
||||||
|
|
||||||
if (isset(self::$operatorHash[$expression[0]])) {
|
|
||||||
$result['operator'] = $expression[0];
|
|
||||||
$expression = substr($expression, 1);
|
|
||||||
} else {
|
|
||||||
$result['operator'] = '';
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (explode(',', $expression) as $value) {
|
|
||||||
$value = trim($value);
|
|
||||||
$varspec = [];
|
|
||||||
if ($colonPos = strpos($value, ':')) {
|
|
||||||
$varspec['value'] = substr($value, 0, $colonPos);
|
|
||||||
$varspec['modifier'] = ':';
|
|
||||||
$varspec['position'] = (int) substr($value, $colonPos + 1);
|
|
||||||
} elseif (substr($value, -1) === '*') {
|
|
||||||
$varspec['modifier'] = '*';
|
|
||||||
$varspec['value'] = substr($value, 0, -1);
|
|
||||||
} else {
|
|
||||||
$varspec['value'] = (string) $value;
|
|
||||||
$varspec['modifier'] = '';
|
|
||||||
}
|
|
||||||
$result['values'][] = $varspec;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Process an expansion
|
|
||||||
*
|
|
||||||
* @param array $matches Matches met in the preg_replace_callback
|
|
||||||
*
|
|
||||||
* @return string Returns the replacement string
|
|
||||||
*/
|
|
||||||
private function expandMatch(array $matches)
|
|
||||||
{
|
|
||||||
static $rfc1738to3986 = ['+' => '%20', '%7e' => '~'];
|
|
||||||
|
|
||||||
$replacements = [];
|
|
||||||
$parsed = self::parseExpression($matches[1]);
|
|
||||||
$prefix = self::$operatorHash[$parsed['operator']]['prefix'];
|
|
||||||
$joiner = self::$operatorHash[$parsed['operator']]['joiner'];
|
|
||||||
$useQuery = self::$operatorHash[$parsed['operator']]['query'];
|
|
||||||
|
|
||||||
foreach ($parsed['values'] as $value) {
|
|
||||||
if (!isset($this->variables[$value['value']])) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$variable = $this->variables[$value['value']];
|
|
||||||
$actuallyUseQuery = $useQuery;
|
|
||||||
$expanded = '';
|
|
||||||
|
|
||||||
if (is_array($variable)) {
|
|
||||||
$isAssoc = $this->isAssoc($variable);
|
|
||||||
$kvp = [];
|
|
||||||
foreach ($variable as $key => $var) {
|
|
||||||
if ($isAssoc) {
|
|
||||||
$key = rawurlencode($key);
|
|
||||||
$isNestedArray = is_array($var);
|
|
||||||
} else {
|
|
||||||
$isNestedArray = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!$isNestedArray) {
|
|
||||||
$var = rawurlencode($var);
|
|
||||||
if ($parsed['operator'] === '+' ||
|
|
||||||
$parsed['operator'] === '#'
|
|
||||||
) {
|
|
||||||
$var = $this->decodeReserved($var);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($value['modifier'] === '*') {
|
|
||||||
if ($isAssoc) {
|
|
||||||
if ($isNestedArray) {
|
|
||||||
// Nested arrays must allow for deeply nested
|
|
||||||
// structures.
|
|
||||||
$var = strtr(
|
|
||||||
http_build_query([$key => $var]),
|
|
||||||
$rfc1738to3986
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
$var = $key . '=' . $var;
|
|
||||||
}
|
|
||||||
} elseif ($key > 0 && $actuallyUseQuery) {
|
|
||||||
$var = $value['value'] . '=' . $var;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$kvp[$key] = $var;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (empty($variable)) {
|
|
||||||
$actuallyUseQuery = false;
|
|
||||||
} elseif ($value['modifier'] === '*') {
|
|
||||||
$expanded = implode($joiner, $kvp);
|
|
||||||
if ($isAssoc) {
|
|
||||||
// Don't prepend the value name when using the explode
|
|
||||||
// modifier with an associative array.
|
|
||||||
$actuallyUseQuery = false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if ($isAssoc) {
|
|
||||||
// When an associative array is encountered and the
|
|
||||||
// explode modifier is not set, then the result must be
|
|
||||||
// a comma separated list of keys followed by their
|
|
||||||
// respective values.
|
|
||||||
foreach ($kvp as $k => &$v) {
|
|
||||||
$v = $k . ',' . $v;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$expanded = implode(',', $kvp);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if ($value['modifier'] === ':') {
|
|
||||||
$variable = substr($variable, 0, $value['position']);
|
|
||||||
}
|
|
||||||
$expanded = rawurlencode($variable);
|
|
||||||
if ($parsed['operator'] === '+' || $parsed['operator'] === '#') {
|
|
||||||
$expanded = $this->decodeReserved($expanded);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($actuallyUseQuery) {
|
|
||||||
if (!$expanded && $joiner !== '&') {
|
|
||||||
$expanded = $value['value'];
|
|
||||||
} else {
|
|
||||||
$expanded = $value['value'] . '=' . $expanded;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$replacements[] = $expanded;
|
|
||||||
}
|
|
||||||
|
|
||||||
$ret = implode($joiner, $replacements);
|
|
||||||
if ($ret && $prefix) {
|
|
||||||
return $prefix . $ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determines if an array is associative.
|
|
||||||
*
|
|
||||||
* This makes the assumption that input arrays are sequences or hashes.
|
|
||||||
* This assumption is a tradeoff for accuracy in favor of speed, but it
|
|
||||||
* should work in almost every case where input is supplied for a URI
|
|
||||||
* template.
|
|
||||||
*
|
|
||||||
* @param array $array Array to check
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
private function isAssoc(array $array)
|
|
||||||
{
|
|
||||||
return $array && array_keys($array)[0] !== 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Removes percent encoding on reserved characters (used with + and #
|
|
||||||
* modifiers).
|
|
||||||
*
|
|
||||||
* @param string $string String to fix
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
private function decodeReserved($string)
|
|
||||||
{
|
|
||||||
return str_replace(self::$delimsPct, self::$delims, $string);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-92
@@ -1,92 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Exception\InvalidArgumentException;
|
|
||||||
use Psr\Http\Message\UriInterface;
|
|
||||||
use Symfony\Polyfill\Intl\Idn\Idn;
|
|
||||||
|
|
||||||
final class Utils
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* Wrapper for the hrtime() or microtime() functions
|
|
||||||
* (depending on the PHP version, one of the two is used)
|
|
||||||
*
|
|
||||||
* @return float|mixed UNIX timestamp
|
|
||||||
*
|
|
||||||
* @internal
|
|
||||||
*/
|
|
||||||
public static function currentTime()
|
|
||||||
{
|
|
||||||
return function_exists('hrtime') ? hrtime(true) / 1e9 : microtime(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param int $options
|
|
||||||
*
|
|
||||||
* @return UriInterface
|
|
||||||
* @throws InvalidArgumentException
|
|
||||||
*
|
|
||||||
* @internal
|
|
||||||
*/
|
|
||||||
public static function idnUriConvert(UriInterface $uri, $options = 0)
|
|
||||||
{
|
|
||||||
if ($uri->getHost()) {
|
|
||||||
$asciiHost = self::idnToAsci($uri->getHost(), $options, $info);
|
|
||||||
if ($asciiHost === false) {
|
|
||||||
$errorBitSet = isset($info['errors']) ? $info['errors'] : 0;
|
|
||||||
|
|
||||||
$errorConstants = array_filter(array_keys(get_defined_constants()), function ($name) {
|
|
||||||
return substr($name, 0, 11) === 'IDNA_ERROR_';
|
|
||||||
});
|
|
||||||
|
|
||||||
$errors = [];
|
|
||||||
foreach ($errorConstants as $errorConstant) {
|
|
||||||
if ($errorBitSet & constant($errorConstant)) {
|
|
||||||
$errors[] = $errorConstant;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$errorMessage = 'IDN conversion failed';
|
|
||||||
if ($errors) {
|
|
||||||
$errorMessage .= ' (errors: ' . implode(', ', $errors) . ')';
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new InvalidArgumentException($errorMessage);
|
|
||||||
} else {
|
|
||||||
if ($uri->getHost() !== $asciiHost) {
|
|
||||||
// Replace URI only if the ASCII version is different
|
|
||||||
$uri = $uri->withHost($asciiHost);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return $uri;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $domain
|
|
||||||
* @param int $options
|
|
||||||
* @param array $info
|
|
||||||
*
|
|
||||||
* @return string|false
|
|
||||||
*/
|
|
||||||
private static function idnToAsci($domain, $options, &$info = [])
|
|
||||||
{
|
|
||||||
if (\preg_match('%^[ -~]+$%', $domain) === 1) {
|
|
||||||
return $domain;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (\extension_loaded('intl') && defined('INTL_IDNA_VARIANT_UTS46')) {
|
|
||||||
return \idn_to_ascii($domain, $options, INTL_IDNA_VARIANT_UTS46, $info);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* The Idn class is marked as @internal. Verify that class and method exists.
|
|
||||||
*/
|
|
||||||
if (method_exists(Idn::class, 'idn_to_ascii')) {
|
|
||||||
return Idn::idn_to_ascii($domain, $options, Idn::INTL_IDNA_VARIANT_UTS46, $info);
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new \RuntimeException('ext-intl or symfony/polyfill-intl-idn not loaded or too old');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-334
@@ -1,334 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp;
|
|
||||||
|
|
||||||
use GuzzleHttp\Handler\CurlHandler;
|
|
||||||
use GuzzleHttp\Handler\CurlMultiHandler;
|
|
||||||
use GuzzleHttp\Handler\Proxy;
|
|
||||||
use GuzzleHttp\Handler\StreamHandler;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Expands a URI template
|
|
||||||
*
|
|
||||||
* @param string $template URI template
|
|
||||||
* @param array $variables Template variables
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function uri_template($template, array $variables)
|
|
||||||
{
|
|
||||||
if (extension_loaded('uri_template')) {
|
|
||||||
// @codeCoverageIgnoreStart
|
|
||||||
return \uri_template($template, $variables);
|
|
||||||
// @codeCoverageIgnoreEnd
|
|
||||||
}
|
|
||||||
|
|
||||||
static $uriTemplate;
|
|
||||||
if (!$uriTemplate) {
|
|
||||||
$uriTemplate = new UriTemplate();
|
|
||||||
}
|
|
||||||
|
|
||||||
return $uriTemplate->expand($template, $variables);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Debug function used to describe the provided value type and class.
|
|
||||||
*
|
|
||||||
* @param mixed $input
|
|
||||||
*
|
|
||||||
* @return string Returns a string containing the type of the variable and
|
|
||||||
* if a class is provided, the class name.
|
|
||||||
*/
|
|
||||||
function describe_type($input)
|
|
||||||
{
|
|
||||||
switch (gettype($input)) {
|
|
||||||
case 'object':
|
|
||||||
return 'object(' . get_class($input) . ')';
|
|
||||||
case 'array':
|
|
||||||
return 'array(' . count($input) . ')';
|
|
||||||
default:
|
|
||||||
ob_start();
|
|
||||||
var_dump($input);
|
|
||||||
// normalize float vs double
|
|
||||||
return str_replace('double(', 'float(', rtrim(ob_get_clean()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Parses an array of header lines into an associative array of headers.
|
|
||||||
*
|
|
||||||
* @param iterable $lines Header lines array of strings in the following
|
|
||||||
* format: "Name: Value"
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
function headers_from_lines($lines)
|
|
||||||
{
|
|
||||||
$headers = [];
|
|
||||||
|
|
||||||
foreach ($lines as $line) {
|
|
||||||
$parts = explode(':', $line, 2);
|
|
||||||
$headers[trim($parts[0])][] = isset($parts[1])
|
|
||||||
? trim($parts[1])
|
|
||||||
: null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $headers;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns a debug stream based on the provided variable.
|
|
||||||
*
|
|
||||||
* @param mixed $value Optional value
|
|
||||||
*
|
|
||||||
* @return resource
|
|
||||||
*/
|
|
||||||
function debug_resource($value = null)
|
|
||||||
{
|
|
||||||
if (is_resource($value)) {
|
|
||||||
return $value;
|
|
||||||
} elseif (defined('STDOUT')) {
|
|
||||||
return STDOUT;
|
|
||||||
}
|
|
||||||
|
|
||||||
return fopen('php://output', 'w');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Chooses and creates a default handler to use based on the environment.
|
|
||||||
*
|
|
||||||
* The returned handler is not wrapped by any default middlewares.
|
|
||||||
*
|
|
||||||
* @return callable Returns the best handler for the given system.
|
|
||||||
* @throws \RuntimeException if no viable Handler is available.
|
|
||||||
*/
|
|
||||||
function choose_handler()
|
|
||||||
{
|
|
||||||
$handler = null;
|
|
||||||
if (function_exists('curl_multi_exec') && function_exists('curl_exec')) {
|
|
||||||
$handler = Proxy::wrapSync(new CurlMultiHandler(), new CurlHandler());
|
|
||||||
} elseif (function_exists('curl_exec')) {
|
|
||||||
$handler = new CurlHandler();
|
|
||||||
} elseif (function_exists('curl_multi_exec')) {
|
|
||||||
$handler = new CurlMultiHandler();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ini_get('allow_url_fopen')) {
|
|
||||||
$handler = $handler
|
|
||||||
? Proxy::wrapStreaming($handler, new StreamHandler())
|
|
||||||
: new StreamHandler();
|
|
||||||
} elseif (!$handler) {
|
|
||||||
throw new \RuntimeException('GuzzleHttp requires cURL, the '
|
|
||||||
. 'allow_url_fopen ini setting, or a custom HTTP handler.');
|
|
||||||
}
|
|
||||||
|
|
||||||
return $handler;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the default User-Agent string to use with Guzzle
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
function default_user_agent()
|
|
||||||
{
|
|
||||||
static $defaultAgent = '';
|
|
||||||
|
|
||||||
if (!$defaultAgent) {
|
|
||||||
$defaultAgent = 'GuzzleHttp/' . Client::VERSION;
|
|
||||||
if (extension_loaded('curl') && function_exists('curl_version')) {
|
|
||||||
$defaultAgent .= ' curl/' . \curl_version()['version'];
|
|
||||||
}
|
|
||||||
$defaultAgent .= ' PHP/' . PHP_VERSION;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $defaultAgent;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the default cacert bundle for the current system.
|
|
||||||
*
|
|
||||||
* First, the openssl.cafile and curl.cainfo php.ini settings are checked.
|
|
||||||
* If those settings are not configured, then the common locations for
|
|
||||||
* bundles found on Red Hat, CentOS, Fedora, Ubuntu, Debian, FreeBSD, OS X
|
|
||||||
* and Windows are checked. If any of these file locations are found on
|
|
||||||
* disk, they will be utilized.
|
|
||||||
*
|
|
||||||
* Note: the result of this function is cached for subsequent calls.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
* @throws \RuntimeException if no bundle can be found.
|
|
||||||
*/
|
|
||||||
function default_ca_bundle()
|
|
||||||
{
|
|
||||||
static $cached = null;
|
|
||||||
static $cafiles = [
|
|
||||||
// Red Hat, CentOS, Fedora (provided by the ca-certificates package)
|
|
||||||
'/etc/pki/tls/certs/ca-bundle.crt',
|
|
||||||
// Ubuntu, Debian (provided by the ca-certificates package)
|
|
||||||
'/etc/ssl/certs/ca-certificates.crt',
|
|
||||||
// FreeBSD (provided by the ca_root_nss package)
|
|
||||||
'/usr/local/share/certs/ca-root-nss.crt',
|
|
||||||
// SLES 12 (provided by the ca-certificates package)
|
|
||||||
'/var/lib/ca-certificates/ca-bundle.pem',
|
|
||||||
// OS X provided by homebrew (using the default path)
|
|
||||||
'/usr/local/etc/openssl/cert.pem',
|
|
||||||
// Google app engine
|
|
||||||
'/etc/ca-certificates.crt',
|
|
||||||
// Windows?
|
|
||||||
'C:\\windows\\system32\\curl-ca-bundle.crt',
|
|
||||||
'C:\\windows\\curl-ca-bundle.crt',
|
|
||||||
];
|
|
||||||
|
|
||||||
if ($cached) {
|
|
||||||
return $cached;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($ca = ini_get('openssl.cafile')) {
|
|
||||||
return $cached = $ca;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($ca = ini_get('curl.cainfo')) {
|
|
||||||
return $cached = $ca;
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach ($cafiles as $filename) {
|
|
||||||
if (file_exists($filename)) {
|
|
||||||
return $cached = $filename;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new \RuntimeException(
|
|
||||||
<<< EOT
|
|
||||||
No system CA bundle could be found in any of the the common system locations.
|
|
||||||
PHP versions earlier than 5.6 are not properly configured to use the system's
|
|
||||||
CA bundle by default. In order to verify peer certificates, you will need to
|
|
||||||
supply the path on disk to a certificate bundle to the 'verify' request
|
|
||||||
option: http://docs.guzzlephp.org/en/latest/clients.html#verify. If you do not
|
|
||||||
need a specific certificate bundle, then Mozilla provides a commonly used CA
|
|
||||||
bundle which can be downloaded here (provided by the maintainer of cURL):
|
|
||||||
https://raw.githubusercontent.com/bagder/ca-bundle/master/ca-bundle.crt. Once
|
|
||||||
you have a CA bundle available on disk, you can set the 'openssl.cafile' PHP
|
|
||||||
ini setting to point to the path to the file, allowing you to omit the 'verify'
|
|
||||||
request option. See http://curl.haxx.se/docs/sslcerts.html for more
|
|
||||||
information.
|
|
||||||
EOT
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates an associative array of lowercase header names to the actual
|
|
||||||
* header casing.
|
|
||||||
*
|
|
||||||
* @param array $headers
|
|
||||||
*
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
function normalize_header_keys(array $headers)
|
|
||||||
{
|
|
||||||
$result = [];
|
|
||||||
foreach (array_keys($headers) as $key) {
|
|
||||||
$result[strtolower($key)] = $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns true if the provided host matches any of the no proxy areas.
|
|
||||||
*
|
|
||||||
* This method will strip a port from the host if it is present. Each pattern
|
|
||||||
* can be matched with an exact match (e.g., "foo.com" == "foo.com") or a
|
|
||||||
* partial match: (e.g., "foo.com" == "baz.foo.com" and ".foo.com" ==
|
|
||||||
* "baz.foo.com", but ".foo.com" != "foo.com").
|
|
||||||
*
|
|
||||||
* Areas are matched in the following cases:
|
|
||||||
* 1. "*" (without quotes) always matches any hosts.
|
|
||||||
* 2. An exact match.
|
|
||||||
* 3. The area starts with "." and the area is the last part of the host. e.g.
|
|
||||||
* '.mit.edu' will match any host that ends with '.mit.edu'.
|
|
||||||
*
|
|
||||||
* @param string $host Host to check against the patterns.
|
|
||||||
* @param array $noProxyArray An array of host patterns.
|
|
||||||
*
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
function is_host_in_noproxy($host, array $noProxyArray)
|
|
||||||
{
|
|
||||||
if (strlen($host) === 0) {
|
|
||||||
throw new \InvalidArgumentException('Empty host provided');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Strip port if present.
|
|
||||||
if (strpos($host, ':')) {
|
|
||||||
$host = explode($host, ':', 2)[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach ($noProxyArray as $area) {
|
|
||||||
// Always match on wildcards.
|
|
||||||
if ($area === '*') {
|
|
||||||
return true;
|
|
||||||
} elseif (empty($area)) {
|
|
||||||
// Don't match on empty values.
|
|
||||||
continue;
|
|
||||||
} elseif ($area === $host) {
|
|
||||||
// Exact matches.
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
// Special match if the area when prefixed with ".". Remove any
|
|
||||||
// existing leading "." and add a new leading ".".
|
|
||||||
$area = '.' . ltrim($area, '.');
|
|
||||||
if (substr($host, -(strlen($area))) === $area) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper for json_decode that throws when an error occurs.
|
|
||||||
*
|
|
||||||
* @param string $json JSON data to parse
|
|
||||||
* @param bool $assoc When true, returned objects will be converted
|
|
||||||
* into associative arrays.
|
|
||||||
* @param int $depth User specified recursion depth.
|
|
||||||
* @param int $options Bitmask of JSON decode options.
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception\InvalidArgumentException if the JSON cannot be decoded.
|
|
||||||
* @link http://www.php.net/manual/en/function.json-decode.php
|
|
||||||
*/
|
|
||||||
function json_decode($json, $assoc = false, $depth = 512, $options = 0)
|
|
||||||
{
|
|
||||||
$data = \json_decode($json, $assoc, $depth, $options);
|
|
||||||
if (JSON_ERROR_NONE !== json_last_error()) {
|
|
||||||
throw new Exception\InvalidArgumentException(
|
|
||||||
'json_decode error: ' . json_last_error_msg()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $data;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper for JSON encoding that throws when an error occurs.
|
|
||||||
*
|
|
||||||
* @param mixed $value The value being encoded
|
|
||||||
* @param int $options JSON encode option bitmask
|
|
||||||
* @param int $depth Set the maximum depth. Must be greater than zero.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
* @throws Exception\InvalidArgumentException if the JSON cannot be encoded.
|
|
||||||
* @link http://www.php.net/manual/en/function.json-encode.php
|
|
||||||
*/
|
|
||||||
function json_encode($value, $options = 0, $depth = 512)
|
|
||||||
{
|
|
||||||
$json = \json_encode($value, $options, $depth);
|
|
||||||
if (JSON_ERROR_NONE !== json_last_error()) {
|
|
||||||
throw new Exception\InvalidArgumentException(
|
|
||||||
'json_encode error: ' . json_last_error_msg()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return $json;
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
// Don't redefine the functions if included multiple times.
|
|
||||||
if (!function_exists('GuzzleHttp\uri_template')) {
|
|
||||||
require __DIR__ . '/functions.php';
|
|
||||||
}
|
|
||||||
-65
@@ -1,65 +0,0 @@
|
|||||||
# CHANGELOG
|
|
||||||
|
|
||||||
|
|
||||||
## 1.3.1 - 2016-12-20
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- `wait()` foreign promise compatibility
|
|
||||||
|
|
||||||
|
|
||||||
## 1.3.0 - 2016-11-18
|
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
- Adds support for custom task queues.
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
|
|
||||||
- Fixed coroutine promise memory leak.
|
|
||||||
|
|
||||||
|
|
||||||
## 1.2.0 - 2016-05-18
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Update to now catch `\Throwable` on PHP 7+
|
|
||||||
|
|
||||||
|
|
||||||
## 1.1.0 - 2016-03-07
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Update EachPromise to prevent recurring on a iterator when advancing, as this
|
|
||||||
could trigger fatal generator errors.
|
|
||||||
- Update Promise to allow recursive waiting without unwrapping exceptions.
|
|
||||||
|
|
||||||
|
|
||||||
## 1.0.3 - 2015-10-15
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Update EachPromise to immediately resolve when the underlying promise iterator
|
|
||||||
is empty. Previously, such a promise would throw an exception when its `wait`
|
|
||||||
function was called.
|
|
||||||
|
|
||||||
|
|
||||||
## 1.0.2 - 2015-05-15
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Conditionally require functions.php.
|
|
||||||
|
|
||||||
|
|
||||||
## 1.0.1 - 2015-06-24
|
|
||||||
|
|
||||||
### Changed
|
|
||||||
|
|
||||||
- Updating EachPromise to call next on the underlying promise iterator as late
|
|
||||||
as possible to ensure that generators that generate new requests based on
|
|
||||||
callbacks are not iterated until after callbacks are invoked.
|
|
||||||
|
|
||||||
|
|
||||||
## 1.0.0 - 2015-05-12
|
|
||||||
|
|
||||||
- Initial release
|
|
||||||
Vendored
-19
@@ -1,19 +0,0 @@
|
|||||||
Copyright (c) 2015-2016 Michael Dowling, https://github.com/mtdowling <mtdowling@gmail.com>
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in
|
|
||||||
all copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
THE SOFTWARE.
|
|
||||||
Vendored
-13
@@ -1,13 +0,0 @@
|
|||||||
all: clean test
|
|
||||||
|
|
||||||
test:
|
|
||||||
vendor/bin/phpunit
|
|
||||||
|
|
||||||
coverage:
|
|
||||||
vendor/bin/phpunit --coverage-html=artifacts/coverage
|
|
||||||
|
|
||||||
view-coverage:
|
|
||||||
open artifacts/coverage/index.html
|
|
||||||
|
|
||||||
clean:
|
|
||||||
rm -rf artifacts/*
|
|
||||||
Vendored
-504
@@ -1,504 +0,0 @@
|
|||||||
# Guzzle Promises
|
|
||||||
|
|
||||||
[Promises/A+](https://promisesaplus.com/) implementation that handles promise
|
|
||||||
chaining and resolution iteratively, allowing for "infinite" promise chaining
|
|
||||||
while keeping the stack size constant. Read [this blog post](https://blog.domenic.me/youre-missing-the-point-of-promises/)
|
|
||||||
for a general introduction to promises.
|
|
||||||
|
|
||||||
- [Features](#features)
|
|
||||||
- [Quick start](#quick-start)
|
|
||||||
- [Synchronous wait](#synchronous-wait)
|
|
||||||
- [Cancellation](#cancellation)
|
|
||||||
- [API](#api)
|
|
||||||
- [Promise](#promise)
|
|
||||||
- [FulfilledPromise](#fulfilledpromise)
|
|
||||||
- [RejectedPromise](#rejectedpromise)
|
|
||||||
- [Promise interop](#promise-interop)
|
|
||||||
- [Implementation notes](#implementation-notes)
|
|
||||||
|
|
||||||
|
|
||||||
# Features
|
|
||||||
|
|
||||||
- [Promises/A+](https://promisesaplus.com/) implementation.
|
|
||||||
- Promise resolution and chaining is handled iteratively, allowing for
|
|
||||||
"infinite" promise chaining.
|
|
||||||
- Promises have a synchronous `wait` method.
|
|
||||||
- Promises can be cancelled.
|
|
||||||
- Works with any object that has a `then` function.
|
|
||||||
- C# style async/await coroutine promises using
|
|
||||||
`GuzzleHttp\Promise\coroutine()`.
|
|
||||||
|
|
||||||
|
|
||||||
# Quick start
|
|
||||||
|
|
||||||
A *promise* represents the eventual result of an asynchronous operation. The
|
|
||||||
primary way of interacting with a promise is through its `then` method, which
|
|
||||||
registers callbacks to receive either a promise's eventual value or the reason
|
|
||||||
why the promise cannot be fulfilled.
|
|
||||||
|
|
||||||
|
|
||||||
## Callbacks
|
|
||||||
|
|
||||||
Callbacks are registered with the `then` method by providing an optional
|
|
||||||
`$onFulfilled` followed by an optional `$onRejected` function.
|
|
||||||
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise->then(
|
|
||||||
// $onFulfilled
|
|
||||||
function ($value) {
|
|
||||||
echo 'The promise was fulfilled.';
|
|
||||||
},
|
|
||||||
// $onRejected
|
|
||||||
function ($reason) {
|
|
||||||
echo 'The promise was rejected.';
|
|
||||||
}
|
|
||||||
);
|
|
||||||
```
|
|
||||||
|
|
||||||
*Resolving* a promise means that you either fulfill a promise with a *value* or
|
|
||||||
reject a promise with a *reason*. Resolving a promises triggers callbacks
|
|
||||||
registered with the promises's `then` method. These callbacks are triggered
|
|
||||||
only once and in the order in which they were added.
|
|
||||||
|
|
||||||
|
|
||||||
## Resolving a promise
|
|
||||||
|
|
||||||
Promises are fulfilled using the `resolve($value)` method. Resolving a promise
|
|
||||||
with any value other than a `GuzzleHttp\Promise\RejectedPromise` will trigger
|
|
||||||
all of the onFulfilled callbacks (resolving a promise with a rejected promise
|
|
||||||
will reject the promise and trigger the `$onRejected` callbacks).
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise
|
|
||||||
->then(function ($value) {
|
|
||||||
// Return a value and don't break the chain
|
|
||||||
return "Hello, " . $value;
|
|
||||||
})
|
|
||||||
// This then is executed after the first then and receives the value
|
|
||||||
// returned from the first then.
|
|
||||||
->then(function ($value) {
|
|
||||||
echo $value;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Resolving the promise triggers the $onFulfilled callbacks and outputs
|
|
||||||
// "Hello, reader".
|
|
||||||
$promise->resolve('reader.');
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
## Promise forwarding
|
|
||||||
|
|
||||||
Promises can be chained one after the other. Each then in the chain is a new
|
|
||||||
promise. The return value of a promise is what's forwarded to the next
|
|
||||||
promise in the chain. Returning a promise in a `then` callback will cause the
|
|
||||||
subsequent promises in the chain to only be fulfilled when the returned promise
|
|
||||||
has been fulfilled. The next promise in the chain will be invoked with the
|
|
||||||
resolved value of the promise.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
|
|
||||||
$promise = new Promise();
|
|
||||||
$nextPromise = new Promise();
|
|
||||||
|
|
||||||
$promise
|
|
||||||
->then(function ($value) use ($nextPromise) {
|
|
||||||
echo $value;
|
|
||||||
return $nextPromise;
|
|
||||||
})
|
|
||||||
->then(function ($value) {
|
|
||||||
echo $value;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Triggers the first callback and outputs "A"
|
|
||||||
$promise->resolve('A');
|
|
||||||
// Triggers the second callback and outputs "B"
|
|
||||||
$nextPromise->resolve('B');
|
|
||||||
```
|
|
||||||
|
|
||||||
## Promise rejection
|
|
||||||
|
|
||||||
When a promise is rejected, the `$onRejected` callbacks are invoked with the
|
|
||||||
rejection reason.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise->then(null, function ($reason) {
|
|
||||||
echo $reason;
|
|
||||||
});
|
|
||||||
|
|
||||||
$promise->reject('Error!');
|
|
||||||
// Outputs "Error!"
|
|
||||||
```
|
|
||||||
|
|
||||||
## Rejection forwarding
|
|
||||||
|
|
||||||
If an exception is thrown in an `$onRejected` callback, subsequent
|
|
||||||
`$onRejected` callbacks are invoked with the thrown exception as the reason.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise->then(null, function ($reason) {
|
|
||||||
throw new \Exception($reason);
|
|
||||||
})->then(null, function ($reason) {
|
|
||||||
assert($reason->getMessage() === 'Error!');
|
|
||||||
});
|
|
||||||
|
|
||||||
$promise->reject('Error!');
|
|
||||||
```
|
|
||||||
|
|
||||||
You can also forward a rejection down the promise chain by returning a
|
|
||||||
`GuzzleHttp\Promise\RejectedPromise` in either an `$onFulfilled` or
|
|
||||||
`$onRejected` callback.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
use GuzzleHttp\Promise\RejectedPromise;
|
|
||||||
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise->then(null, function ($reason) {
|
|
||||||
return new RejectedPromise($reason);
|
|
||||||
})->then(null, function ($reason) {
|
|
||||||
assert($reason === 'Error!');
|
|
||||||
});
|
|
||||||
|
|
||||||
$promise->reject('Error!');
|
|
||||||
```
|
|
||||||
|
|
||||||
If an exception is not thrown in a `$onRejected` callback and the callback
|
|
||||||
does not return a rejected promise, downstream `$onFulfilled` callbacks are
|
|
||||||
invoked using the value returned from the `$onRejected` callback.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
use GuzzleHttp\Promise\RejectedPromise;
|
|
||||||
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise
|
|
||||||
->then(null, function ($reason) {
|
|
||||||
return "It's ok";
|
|
||||||
})
|
|
||||||
->then(function ($value) {
|
|
||||||
assert($value === "It's ok");
|
|
||||||
});
|
|
||||||
|
|
||||||
$promise->reject('Error!');
|
|
||||||
```
|
|
||||||
|
|
||||||
# Synchronous wait
|
|
||||||
|
|
||||||
You can synchronously force promises to complete using a promise's `wait`
|
|
||||||
method. When creating a promise, you can provide a wait function that is used
|
|
||||||
to synchronously force a promise to complete. When a wait function is invoked
|
|
||||||
it is expected to deliver a value to the promise or reject the promise. If the
|
|
||||||
wait function does not deliver a value, then an exception is thrown. The wait
|
|
||||||
function provided to a promise constructor is invoked when the `wait` function
|
|
||||||
of the promise is called.
|
|
||||||
|
|
||||||
```php
|
|
||||||
$promise = new Promise(function () use (&$promise) {
|
|
||||||
$promise->resolve('foo');
|
|
||||||
});
|
|
||||||
|
|
||||||
// Calling wait will return the value of the promise.
|
|
||||||
echo $promise->wait(); // outputs "foo"
|
|
||||||
```
|
|
||||||
|
|
||||||
If an exception is encountered while invoking the wait function of a promise,
|
|
||||||
the promise is rejected with the exception and the exception is thrown.
|
|
||||||
|
|
||||||
```php
|
|
||||||
$promise = new Promise(function () use (&$promise) {
|
|
||||||
throw new \Exception('foo');
|
|
||||||
});
|
|
||||||
|
|
||||||
$promise->wait(); // throws the exception.
|
|
||||||
```
|
|
||||||
|
|
||||||
Calling `wait` on a promise that has been fulfilled will not trigger the wait
|
|
||||||
function. It will simply return the previously resolved value.
|
|
||||||
|
|
||||||
```php
|
|
||||||
$promise = new Promise(function () { die('this is not called!'); });
|
|
||||||
$promise->resolve('foo');
|
|
||||||
echo $promise->wait(); // outputs "foo"
|
|
||||||
```
|
|
||||||
|
|
||||||
Calling `wait` on a promise that has been rejected will throw an exception. If
|
|
||||||
the rejection reason is an instance of `\Exception` the reason is thrown.
|
|
||||||
Otherwise, a `GuzzleHttp\Promise\RejectionException` is thrown and the reason
|
|
||||||
can be obtained by calling the `getReason` method of the exception.
|
|
||||||
|
|
||||||
```php
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise->reject('foo');
|
|
||||||
$promise->wait();
|
|
||||||
```
|
|
||||||
|
|
||||||
> PHP Fatal error: Uncaught exception 'GuzzleHttp\Promise\RejectionException' with message 'The promise was rejected with value: foo'
|
|
||||||
|
|
||||||
|
|
||||||
## Unwrapping a promise
|
|
||||||
|
|
||||||
When synchronously waiting on a promise, you are joining the state of the
|
|
||||||
promise into the current state of execution (i.e., return the value of the
|
|
||||||
promise if it was fulfilled or throw an exception if it was rejected). This is
|
|
||||||
called "unwrapping" the promise. Waiting on a promise will by default unwrap
|
|
||||||
the promise state.
|
|
||||||
|
|
||||||
You can force a promise to resolve and *not* unwrap the state of the promise
|
|
||||||
by passing `false` to the first argument of the `wait` function:
|
|
||||||
|
|
||||||
```php
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise->reject('foo');
|
|
||||||
// This will not throw an exception. It simply ensures the promise has
|
|
||||||
// been resolved.
|
|
||||||
$promise->wait(false);
|
|
||||||
```
|
|
||||||
|
|
||||||
When unwrapping a promise, the resolved value of the promise will be waited
|
|
||||||
upon until the unwrapped value is not a promise. This means that if you resolve
|
|
||||||
promise A with a promise B and unwrap promise A, the value returned by the
|
|
||||||
wait function will be the value delivered to promise B.
|
|
||||||
|
|
||||||
**Note**: when you do not unwrap the promise, no value is returned.
|
|
||||||
|
|
||||||
|
|
||||||
# Cancellation
|
|
||||||
|
|
||||||
You can cancel a promise that has not yet been fulfilled using the `cancel()`
|
|
||||||
method of a promise. When creating a promise you can provide an optional
|
|
||||||
cancel function that when invoked cancels the action of computing a resolution
|
|
||||||
of the promise.
|
|
||||||
|
|
||||||
|
|
||||||
# API
|
|
||||||
|
|
||||||
|
|
||||||
## Promise
|
|
||||||
|
|
||||||
When creating a promise object, you can provide an optional `$waitFn` and
|
|
||||||
`$cancelFn`. `$waitFn` is a function that is invoked with no arguments and is
|
|
||||||
expected to resolve the promise. `$cancelFn` is a function with no arguments
|
|
||||||
that is expected to cancel the computation of a promise. It is invoked when the
|
|
||||||
`cancel()` method of a promise is called.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
|
|
||||||
$promise = new Promise(
|
|
||||||
function () use (&$promise) {
|
|
||||||
$promise->resolve('waited');
|
|
||||||
},
|
|
||||||
function () {
|
|
||||||
// do something that will cancel the promise computation (e.g., close
|
|
||||||
// a socket, cancel a database query, etc...)
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
assert('waited' === $promise->wait());
|
|
||||||
```
|
|
||||||
|
|
||||||
A promise has the following methods:
|
|
||||||
|
|
||||||
- `then(callable $onFulfilled, callable $onRejected) : PromiseInterface`
|
|
||||||
|
|
||||||
Appends fulfillment and rejection handlers to the promise, and returns a new promise resolving to the return value of the called handler.
|
|
||||||
|
|
||||||
- `otherwise(callable $onRejected) : PromiseInterface`
|
|
||||||
|
|
||||||
Appends a rejection handler callback to the promise, and returns a new promise resolving to the return value of the callback if it is called, or to its original fulfillment value if the promise is instead fulfilled.
|
|
||||||
|
|
||||||
- `wait($unwrap = true) : mixed`
|
|
||||||
|
|
||||||
Synchronously waits on the promise to complete.
|
|
||||||
|
|
||||||
`$unwrap` controls whether or not the value of the promise is returned for a
|
|
||||||
fulfilled promise or if an exception is thrown if the promise is rejected.
|
|
||||||
This is set to `true` by default.
|
|
||||||
|
|
||||||
- `cancel()`
|
|
||||||
|
|
||||||
Attempts to cancel the promise if possible. The promise being cancelled and
|
|
||||||
the parent most ancestor that has not yet been resolved will also be
|
|
||||||
cancelled. Any promises waiting on the cancelled promise to resolve will also
|
|
||||||
be cancelled.
|
|
||||||
|
|
||||||
- `getState() : string`
|
|
||||||
|
|
||||||
Returns the state of the promise. One of `pending`, `fulfilled`, or
|
|
||||||
`rejected`.
|
|
||||||
|
|
||||||
- `resolve($value)`
|
|
||||||
|
|
||||||
Fulfills the promise with the given `$value`.
|
|
||||||
|
|
||||||
- `reject($reason)`
|
|
||||||
|
|
||||||
Rejects the promise with the given `$reason`.
|
|
||||||
|
|
||||||
|
|
||||||
## FulfilledPromise
|
|
||||||
|
|
||||||
A fulfilled promise can be created to represent a promise that has been
|
|
||||||
fulfilled.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\FulfilledPromise;
|
|
||||||
|
|
||||||
$promise = new FulfilledPromise('value');
|
|
||||||
|
|
||||||
// Fulfilled callbacks are immediately invoked.
|
|
||||||
$promise->then(function ($value) {
|
|
||||||
echo $value;
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
## RejectedPromise
|
|
||||||
|
|
||||||
A rejected promise can be created to represent a promise that has been
|
|
||||||
rejected.
|
|
||||||
|
|
||||||
```php
|
|
||||||
use GuzzleHttp\Promise\RejectedPromise;
|
|
||||||
|
|
||||||
$promise = new RejectedPromise('Error');
|
|
||||||
|
|
||||||
// Rejected callbacks are immediately invoked.
|
|
||||||
$promise->then(null, function ($reason) {
|
|
||||||
echo $reason;
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
# Promise interop
|
|
||||||
|
|
||||||
This library works with foreign promises that have a `then` method. This means
|
|
||||||
you can use Guzzle promises with [React promises](https://github.com/reactphp/promise)
|
|
||||||
for example. When a foreign promise is returned inside of a then method
|
|
||||||
callback, promise resolution will occur recursively.
|
|
||||||
|
|
||||||
```php
|
|
||||||
// Create a React promise
|
|
||||||
$deferred = new React\Promise\Deferred();
|
|
||||||
$reactPromise = $deferred->promise();
|
|
||||||
|
|
||||||
// Create a Guzzle promise that is fulfilled with a React promise.
|
|
||||||
$guzzlePromise = new \GuzzleHttp\Promise\Promise();
|
|
||||||
$guzzlePromise->then(function ($value) use ($reactPromise) {
|
|
||||||
// Do something something with the value...
|
|
||||||
// Return the React promise
|
|
||||||
return $reactPromise;
|
|
||||||
});
|
|
||||||
```
|
|
||||||
|
|
||||||
Please note that wait and cancel chaining is no longer possible when forwarding
|
|
||||||
a foreign promise. You will need to wrap a third-party promise with a Guzzle
|
|
||||||
promise in order to utilize wait and cancel functions with foreign promises.
|
|
||||||
|
|
||||||
|
|
||||||
## Event Loop Integration
|
|
||||||
|
|
||||||
In order to keep the stack size constant, Guzzle promises are resolved
|
|
||||||
asynchronously using a task queue. When waiting on promises synchronously, the
|
|
||||||
task queue will be automatically run to ensure that the blocking promise and
|
|
||||||
any forwarded promises are resolved. When using promises asynchronously in an
|
|
||||||
event loop, you will need to run the task queue on each tick of the loop. If
|
|
||||||
you do not run the task queue, then promises will not be resolved.
|
|
||||||
|
|
||||||
You can run the task queue using the `run()` method of the global task queue
|
|
||||||
instance.
|
|
||||||
|
|
||||||
```php
|
|
||||||
// Get the global task queue
|
|
||||||
$queue = \GuzzleHttp\Promise\queue();
|
|
||||||
$queue->run();
|
|
||||||
```
|
|
||||||
|
|
||||||
For example, you could use Guzzle promises with React using a periodic timer:
|
|
||||||
|
|
||||||
```php
|
|
||||||
$loop = React\EventLoop\Factory::create();
|
|
||||||
$loop->addPeriodicTimer(0, [$queue, 'run']);
|
|
||||||
```
|
|
||||||
|
|
||||||
*TODO*: Perhaps adding a `futureTick()` on each tick would be faster?
|
|
||||||
|
|
||||||
|
|
||||||
# Implementation notes
|
|
||||||
|
|
||||||
|
|
||||||
## Promise resolution and chaining is handled iteratively
|
|
||||||
|
|
||||||
By shuffling pending handlers from one owner to another, promises are
|
|
||||||
resolved iteratively, allowing for "infinite" then chaining.
|
|
||||||
|
|
||||||
```php
|
|
||||||
<?php
|
|
||||||
require 'vendor/autoload.php';
|
|
||||||
|
|
||||||
use GuzzleHttp\Promise\Promise;
|
|
||||||
|
|
||||||
$parent = new Promise();
|
|
||||||
$p = $parent;
|
|
||||||
|
|
||||||
for ($i = 0; $i < 1000; $i++) {
|
|
||||||
$p = $p->then(function ($v) {
|
|
||||||
// The stack size remains constant (a good thing)
|
|
||||||
echo xdebug_get_stack_depth() . ', ';
|
|
||||||
return $v + 1;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
$parent->resolve(0);
|
|
||||||
var_dump($p->wait()); // int(1000)
|
|
||||||
|
|
||||||
```
|
|
||||||
|
|
||||||
When a promise is fulfilled or rejected with a non-promise value, the promise
|
|
||||||
then takes ownership of the handlers of each child promise and delivers values
|
|
||||||
down the chain without using recursion.
|
|
||||||
|
|
||||||
When a promise is resolved with another promise, the original promise transfers
|
|
||||||
all of its pending handlers to the new promise. When the new promise is
|
|
||||||
eventually resolved, all of the pending handlers are delivered the forwarded
|
|
||||||
value.
|
|
||||||
|
|
||||||
|
|
||||||
## A promise is the deferred.
|
|
||||||
|
|
||||||
Some promise libraries implement promises using a deferred object to represent
|
|
||||||
a computation and a promise object to represent the delivery of the result of
|
|
||||||
the computation. This is a nice separation of computation and delivery because
|
|
||||||
consumers of the promise cannot modify the value that will be eventually
|
|
||||||
delivered.
|
|
||||||
|
|
||||||
One side effect of being able to implement promise resolution and chaining
|
|
||||||
iteratively is that you need to be able for one promise to reach into the state
|
|
||||||
of another promise to shuffle around ownership of handlers. In order to achieve
|
|
||||||
this without making the handlers of a promise publicly mutable, a promise is
|
|
||||||
also the deferred value, allowing promises of the same parent class to reach
|
|
||||||
into and modify the private properties of promises of the same type. While this
|
|
||||||
does allow consumers of the value to modify the resolution or rejection of the
|
|
||||||
deferred, it is a small price to pay for keeping the stack size constant.
|
|
||||||
|
|
||||||
```php
|
|
||||||
$promise = new Promise();
|
|
||||||
$promise->then(function ($value) { echo $value; });
|
|
||||||
// The promise is the deferred value, so you can deliver a value to it.
|
|
||||||
$promise->resolve('foo');
|
|
||||||
// prints "foo"
|
|
||||||
```
|
|
||||||
-34
@@ -1,34 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "guzzlehttp/promises",
|
|
||||||
"description": "Guzzle promises library",
|
|
||||||
"keywords": ["promise"],
|
|
||||||
"license": "MIT",
|
|
||||||
"authors": [
|
|
||||||
{
|
|
||||||
"name": "Michael Dowling",
|
|
||||||
"email": "mtdowling@gmail.com",
|
|
||||||
"homepage": "https://github.com/mtdowling"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"require": {
|
|
||||||
"php": ">=5.5.0"
|
|
||||||
},
|
|
||||||
"require-dev": {
|
|
||||||
"phpunit/phpunit": "^4.0"
|
|
||||||
},
|
|
||||||
"autoload": {
|
|
||||||
"psr-4": {
|
|
||||||
"GuzzleHttp\\Promise\\": "src/"
|
|
||||||
},
|
|
||||||
"files": ["src/functions_include.php"]
|
|
||||||
},
|
|
||||||
"scripts": {
|
|
||||||
"test": "vendor/bin/phpunit",
|
|
||||||
"test-ci": "vendor/bin/phpunit --coverage-text"
|
|
||||||
},
|
|
||||||
"extra": {
|
|
||||||
"branch-alias": {
|
|
||||||
"dev-master": "1.4-dev"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Promise;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exception thrown when too many errors occur in the some() or any() methods.
|
|
||||||
*/
|
|
||||||
class AggregateException extends RejectionException
|
|
||||||
{
|
|
||||||
public function __construct($msg, array $reasons)
|
|
||||||
{
|
|
||||||
parent::__construct(
|
|
||||||
$reasons,
|
|
||||||
sprintf('%s; %d rejected promises', $msg, count($reasons))
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
<?php
|
|
||||||
namespace GuzzleHttp\Promise;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Exception that is set as the reason for a promise that has been cancelled.
|
|
||||||
*/
|
|
||||||
class CancellationException extends RejectionException
|
|
||||||
{
|
|
||||||
}
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user