类文档

Jar

💡 云策文档标注

概述

Jar 类是 WordPress 中用于管理 Cookie 的容器对象,实现了 ArrayAccess 和 IteratorAggregate 接口,提供 Cookie 的存储、序列化和请求处理功能。

关键要点

  • Jar 类继承自 ArrayAccess 和 IteratorAggregate,允许通过数组方式访问和迭代 Cookie 数据。
  • 核心方法包括 __construct 初始化、normalize_cookie 标准化 Cookie 数据、register 注册钩子到请求系统。
  • 通过 before_request 方法在请求前自动添加 Cookie 头,before_redirect_check 方法处理响应中的 Cookie。
  • 支持 Cookie 的过期检查和域名匹配,确保符合 RFC 6265 标准。

代码示例

public function __construct($cookies = []) {
    if (is_array($cookies) === false) {
        throw InvalidArgument::create(1, '$cookies', 'array', gettype($cookies));
    }

    $this->cookies = $cookies;
}

public function normalize_cookie($cookie, $key = '') {
    if ($cookie instanceof Cookie) {
        return $cookie;
    }

    return Cookie::parse($cookie, $key);
}

public function before_request($url, &$headers, &$data, &$type, &$options) {
    if (!$url instanceof Iri) {
        $url = new Iri($url);
    }

    if (!empty($this->cookies)) {
        $cookies = [];
        foreach ($this->cookies as $key => $cookie) {
            $cookie = $this->normalize_cookie($cookie, $key);

            // Skip expired cookies
            if ($cookie->is_expired()) {
                continue;
            }

            if ($cookie->domain_matches($url->host)) {
                $cookies[] = $cookie->format_for_header();
            }
        }

        $headers['Cookie'] = implode('; ', $cookies);
    }
}

📄 原文内容

Cookie holder object

Methods

Name Description
Jar::__construct Create a new jar
Jar::before_redirect_check Parse all cookies from a response and attach them to the response
Jar::before_request Add Cookie header to a request if we have any
Jar::getIterator
Jar::normalize_cookie Normalise cookie data into a WpOrgRequestsCookie
Jar::offsetExists
Jar::offsetGet
Jar::offsetSet
Jar::offsetUnset
Jar::register Register the cookie handler with the request’s hooking system

Source

class Jar implements ArrayAccess, IteratorAggregate {
	/**
	 * Actual item data
	 *
	 * @var array
	 */
	protected $cookies = [];

	/**
	 * Create a new jar
	 *
	 * @param array $cookies Existing cookie values
	 *
	 * @throws WpOrgRequestsExceptionInvalidArgument When the passed argument is not an array.
	 */
	public function __construct($cookies = []) {
		if (is_array($cookies) === false) {
			throw InvalidArgument::create(1, '$cookies', 'array', gettype($cookies));
		}

		$this->cookies = $cookies;
	}

	/**
	 * Normalise cookie data into a WpOrgRequestsCookie
	 *
	 * @param string|WpOrgRequestsCookie $cookie Cookie header value, possibly pre-parsed (object).
	 * @param string                        $key    Optional. The name for this cookie.
	 * @return WpOrgRequestsCookie
	 */
	public function normalize_cookie($cookie, $key = '') {
		if ($cookie instanceof Cookie) {
			return $cookie;
		}

		return Cookie::parse($cookie, $key);
	}

	/**
	 * Check if the given item exists
	 *
	 * @param string $offset Item key
	 * @return boolean Does the item exist?
	 */
	#[ReturnTypeWillChange]
	public function offsetExists($offset) {
		return isset($this->cookies[$offset]);
	}

	/**
	 * Get the value for the item
	 *
	 * @param string $offset Item key
	 * @return string|null Item value (null if offsetExists is false)
	 */
	#[ReturnTypeWillChange]
	public function offsetGet($offset) {
		if (!isset($this->cookies[$offset])) {
			return null;
		}

		return $this->cookies[$offset];
	}

	/**
	 * Set the given item
	 *
	 * @param string $offset Item name
	 * @param string $value Item value
	 *
	 * @throws WpOrgRequestsException On attempting to use dictionary as list (`invalidset`)
	 */
	#[ReturnTypeWillChange]
	public function offsetSet($offset, $value) {
		if ($offset === null) {
			throw new Exception('Object is a dictionary, not a list', 'invalidset');
		}

		$this->cookies[$offset] = $value;
	}

	/**
	 * Unset the given header
	 *
	 * @param string $offset The key for the item to unset.
	 */
	#[ReturnTypeWillChange]
	public function offsetUnset($offset) {
		unset($this->cookies[$offset]);
	}

	/**
	 * Get an iterator for the data
	 *
	 * @return ArrayIterator
	 */
	#[ReturnTypeWillChange]
	public function getIterator() {
		return new ArrayIterator($this->cookies);
	}

	/**
	 * Register the cookie handler with the request's hooking system
	 *
	 * @param WpOrgRequestsHookManager $hooks Hooking system
	 */
	public function register(HookManager $hooks) {
		$hooks->register('requests.before_request', [$this, 'before_request']);
		$hooks->register('requests.before_redirect_check', [$this, 'before_redirect_check']);
	}

	/**
	 * Add Cookie header to a request if we have any
	 *
	 * As per RFC 6265, cookies are separated by '; '
	 *
	 * @param string $url
	 * @param array $headers
	 * @param array $data
	 * @param string $type
	 * @param array $options
	 */
	public function before_request($url, &$headers, &$data, &$type, &$options) {
		if (!$url instanceof Iri) {
			$url = new Iri($url);
		}

		if (!empty($this->cookies)) {
			$cookies = [];
			foreach ($this->cookies as $key => $cookie) {
				$cookie = $this->normalize_cookie($cookie, $key);

				// Skip expired cookies
				if ($cookie->is_expired()) {
					continue;
				}

				if ($cookie->domain_matches($url->host)) {
					$cookies[] = $cookie->format_for_header();
				}
			}

			$headers['Cookie'] = implode('; ', $cookies);
		}
	}

	/**
	 * Parse all cookies from a response and attach them to the response
	 *
	 * @param WpOrgRequestsResponse $response Response as received.
	 */
	public function before_redirect_check(Response $response) {
		$url = $response->url;
		if (!$url instanceof Iri) {
			$url = new Iri($url);
		}

		$cookies           = Cookie::parse_from_headers($response->headers, $url);
		$this->cookies     = array_merge($this->cookies, $cookies);
		$response->cookies = $this;
	}
}