├── .all-contributorsrc
├── .codeclimate.yml
├── .editorconfig
├── .gitignore
├── .travis.yml
├── LICENSE.md
├── README.md
├── composer.json
├── includes
├── functions.php
└── object-cache.php
├── package.json
├── phpcs.xml
├── phpmd.xml
├── phpunit.xml
├── src
├── CacheAdapter
│ ├── CacheAdapter.php
│ ├── CacheAdapterFactory.php
│ ├── Psr16CacheAdapter.php
│ ├── Psr6CacheAdapter.php
│ └── PsrCacheAdapterFactory.php
├── CacheKeyGen
│ ├── CacheKeyGen.php
│ ├── WpCacheKeyGen.php
│ └── WpPsrCacheKeyGen.php
├── CacheSelector
│ ├── BaseCacheSelector.php
│ ├── BaseCacheSelectorFactory.php
│ ├── CacheSelector.php
│ └── CacheSelectorFactory.php
├── ObjectCache.php
├── ObjectCacheFactory.php
└── ObjectCacheService.php
└── tests
└── KeyGenTest.php
/.all-contributorsrc:
--------------------------------------------------------------------------------
1 | {
2 | "projectName": "wp-psr-cache",
3 | "projectOwner": "felixarntz",
4 | "files": [
5 | "README.md"
6 | ],
7 | "imageSize": 100,
8 | "commit": false,
9 | "contributors": [
10 | {
11 | "login": "felixarntz",
12 | "name": "Felix Arntz",
13 | "avatar_url": "https://avatars1.githubusercontent.com/u/3531426?v=4",
14 | "profile": "https://leaves-and-love.net",
15 | "contributions": [
16 | "code",
17 | "bug",
18 | "doc",
19 | "example",
20 | "ideas",
21 | "test"
22 | ]
23 | },
24 | {
25 | "login": "schlessera",
26 | "name": "Alain Schlesser",
27 | "avatar_url": "https://avatars1.githubusercontent.com/u/83631?v=4",
28 | "profile": "https://www.alainschlesser.com/",
29 | "contributions": [
30 | "code",
31 | "bug",
32 | "review"
33 | ]
34 | },
35 | {
36 | "login": "tfrommen",
37 | "name": "Thorsten Frommen",
38 | "avatar_url": "https://avatars2.githubusercontent.com/u/6049306?v=4",
39 | "profile": "https://tfrommen.de",
40 | "contributions": [
41 | "review"
42 | ]
43 | },
44 | {
45 | "login": "moorscode",
46 | "name": "Jip",
47 | "avatar_url": "https://avatars0.githubusercontent.com/u/2005352?v=4",
48 | "profile": "http://www.jipmoors.nl",
49 | "contributions": [
50 | "ideas"
51 | ]
52 | }
53 | ]
54 | }
55 |
--------------------------------------------------------------------------------
/.codeclimate.yml:
--------------------------------------------------------------------------------
1 | ---
2 | engines:
3 | duplication:
4 | enabled: true
5 | config:
6 | languages:
7 | - php
8 | fixme:
9 | enabled: true
10 | phpcodesniffer:
11 | enabled: true
12 | config:
13 | file_extensions: "php"
14 | standard: "phpcs.xml"
15 | ignore_warnings: true
16 | encoding: utf-8
17 | phpmd:
18 | enabled: true
19 | config:
20 | file_extensions: "php"
21 | rulesets: "phpmd.xml"
22 | ratings:
23 | paths:
24 | - "src/**.php"
25 | exclude_paths:
26 | - "includes/*"
27 | - "tests/*"
28 | - "vendor/*"
--------------------------------------------------------------------------------
/.editorconfig:
--------------------------------------------------------------------------------
1 | # PHP PSR-2 Coding Standards
2 | # http://www.php-fig.org/psr/psr-2/
3 |
4 | root = true
5 |
6 | [*.php]
7 | charset = utf-8
8 | end_of_line = lf
9 | insert_final_newline = true
10 | trim_trailing_whitespace = true
11 | indent_style = space
12 | indent_size = 4
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | #############
2 | ## IDEs
3 | #############
4 |
5 | *.pydevproject
6 | .project
7 | .metadata
8 | *.swp
9 | *~.nib
10 | local.properties
11 | .classpath
12 | .settings/
13 | .loadpath
14 | .externalToolBuilders/
15 | *.launch
16 | .cproject
17 | .buildpath
18 | nbproject/
19 |
20 | #############
21 | ## 3rd-Party
22 | #############
23 |
24 | node_modules/
25 | vendor/
26 |
27 | #############
28 | ## OSes
29 | #############
30 |
31 | [Tt]humbs.db
32 | [Dd]esktop.ini
33 | *.DS_store
34 | .DS_store?
35 |
36 | #############
37 | ## Misc
38 | #############
39 |
40 | tmp/
41 | *.tmp
42 | *.bak
43 | *.log
44 | *.lock
45 | *-lock.json
46 | *.[Cc]ache
47 | *.cpr
48 | *.orig
49 | *.php.in
50 | .idea/
51 | .sass-cache/*
52 | temp/
53 | ._*
54 | .Trashes
55 | .svn
56 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | sudo: false
2 | language: php
3 | cache:
4 | directories:
5 | - vendor
6 | - $HOME/.composer/cache
7 | matrix:
8 | include:
9 | - php: 7.2
10 | env: PHPLINT=1 PHPCS=1 COVERAGE=1
11 | - php: 7.1
12 | - php: 7.0
13 | allow_failures:
14 | - php: nightly
15 | before_script:
16 | - composer self-update
17 | - composer install
18 | - |
19 | if [[ "$COVERAGE" == "1" ]]; then
20 | curl -L https://codeclimate.com/downloads/test-reporter/test-reporter-latest-linux-amd64 > ./cc-test-reporter
21 | chmod +x ./cc-test-reporter
22 | ./cc-test-reporter before-build
23 | fi
24 | script:
25 | - |
26 | if [[ "$PHPLINT" == "1" ]]; then
27 | find -L . -path ./vendor -prune -o -name '*.php' -print0 | xargs -0 -n 1 -P 4 php -l
28 | fi
29 | - |
30 | if [[ "$PHPCS" == "1" ]]; then
31 | vendor/bin/phpcs -v --runtime-set ignore_warnings_on_exit 1
32 | fi
33 | - |
34 | if [[ -z "$CODECLIMATE_REPO_TOKEN" ]]; then
35 | COVERAGE="0"
36 | fi
37 | - |
38 | if [[ "$COVERAGE" == "1" ]]; then
39 | mkdir -p build/logs
40 | vendor/bin/phpunit -c phpunit.xml --coverage-clover build/logs/clover.xml
41 | else
42 | vendor/bin/phpunit -c phpunit.xml
43 | fi
44 | after_script:
45 | - |
46 | if [[ "$COVERAGE" == "1" ]]; then
47 | ./cc-test-reporter after-build --exit-code $TRAVIS_TEST_RESULT
48 | fi
49 | notifications:
50 | email: false
51 |
--------------------------------------------------------------------------------
/LICENSE.md:
--------------------------------------------------------------------------------
1 | ### GNU GENERAL PUBLIC LICENSE
2 |
3 | Version 2, June 1991
4 |
5 | Copyright (C) 1989, 1991 Free Software Foundation, Inc.
6 | 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
7 |
8 | Everyone is permitted to copy and distribute verbatim copies
9 | of this license document, but changing it is not allowed.
10 |
11 | ### Preamble
12 |
13 | The licenses for most software are designed to take away your freedom
14 | to share and change it. By contrast, the GNU General Public License is
15 | intended to guarantee your freedom to share and change free
16 | software--to make sure the software is free for all its users. This
17 | General Public License applies to most of the Free Software
18 | Foundation's software and to any other program whose authors commit to
19 | using it. (Some other Free Software Foundation software is covered by
20 | the GNU Lesser General Public License instead.) You can apply it to
21 | your programs, too.
22 |
23 | When we speak of free software, we are referring to freedom, not
24 | price. Our General Public Licenses are designed to make sure that you
25 | have the freedom to distribute copies of free software (and charge for
26 | this service if you wish), that you receive source code or can get it
27 | if you want it, that you can change the software or use pieces of it
28 | in new free programs; and that you know you can do these things.
29 |
30 | To protect your rights, we need to make restrictions that forbid
31 | anyone to deny you these rights or to ask you to surrender the rights.
32 | These restrictions translate to certain responsibilities for you if
33 | you distribute copies of the software, or if you modify it.
34 |
35 | For example, if you distribute copies of such a program, whether
36 | gratis or for a fee, you must give the recipients all the rights that
37 | you have. You must make sure that they, too, receive or can get the
38 | source code. And you must show them these terms so they know their
39 | rights.
40 |
41 | We protect your rights with two steps: (1) copyright the software, and
42 | (2) offer you this license which gives you legal permission to copy,
43 | distribute and/or modify the software.
44 |
45 | Also, for each author's protection and ours, we want to make certain
46 | that everyone understands that there is no warranty for this free
47 | software. If the software is modified by someone else and passed on,
48 | we want its recipients to know that what they have is not the
49 | original, so that any problems introduced by others will not reflect
50 | on the original authors' reputations.
51 |
52 | Finally, any free program is threatened constantly by software
53 | patents. We wish to avoid the danger that redistributors of a free
54 | program will individually obtain patent licenses, in effect making the
55 | program proprietary. To prevent this, we have made it clear that any
56 | patent must be licensed for everyone's free use or not licensed at
57 | all.
58 |
59 | The precise terms and conditions for copying, distribution and
60 | modification follow.
61 |
62 | ### TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
63 |
64 | **0.** This License applies to any program or other work which
65 | contains a notice placed by the copyright holder saying it may be
66 | distributed under the terms of this General Public License. The
67 | "Program", below, refers to any such program or work, and a "work
68 | based on the Program" means either the Program or any derivative work
69 | under copyright law: that is to say, a work containing the Program or
70 | a portion of it, either verbatim or with modifications and/or
71 | translated into another language. (Hereinafter, translation is
72 | included without limitation in the term "modification".) Each licensee
73 | is addressed as "you".
74 |
75 | Activities other than copying, distribution and modification are not
76 | covered by this License; they are outside its scope. The act of
77 | running the Program is not restricted, and the output from the Program
78 | is covered only if its contents constitute a work based on the Program
79 | (independent of having been made by running the Program). Whether that
80 | is true depends on what the Program does.
81 |
82 | **1.** You may copy and distribute verbatim copies of the Program's
83 | source code as you receive it, in any medium, provided that you
84 | conspicuously and appropriately publish on each copy an appropriate
85 | copyright notice and disclaimer of warranty; keep intact all the
86 | notices that refer to this License and to the absence of any warranty;
87 | and give any other recipients of the Program a copy of this License
88 | along with the Program.
89 |
90 | You may charge a fee for the physical act of transferring a copy, and
91 | you may at your option offer warranty protection in exchange for a
92 | fee.
93 |
94 | **2.** You may modify your copy or copies of the Program or any
95 | portion of it, thus forming a work based on the Program, and copy and
96 | distribute such modifications or work under the terms of Section 1
97 | above, provided that you also meet all of these conditions:
98 |
99 |
100 | **a)** You must cause the modified files to carry prominent notices
101 | stating that you changed the files and the date of any change.
102 |
103 |
104 | **b)** You must cause any work that you distribute or publish, that in
105 | whole or in part contains or is derived from the Program or any part
106 | thereof, to be licensed as a whole at no charge to all third parties
107 | under the terms of this License.
108 |
109 |
110 | **c)** If the modified program normally reads commands interactively
111 | when run, you must cause it, when started running for such interactive
112 | use in the most ordinary way, to print or display an announcement
113 | including an appropriate copyright notice and a notice that there is
114 | no warranty (or else, saying that you provide a warranty) and that
115 | users may redistribute the program under these conditions, and telling
116 | the user how to view a copy of this License. (Exception: if the
117 | Program itself is interactive but does not normally print such an
118 | announcement, your work based on the Program is not required to print
119 | an announcement.)
120 |
121 | These requirements apply to the modified work as a whole. If
122 | identifiable sections of that work are not derived from the Program,
123 | and can be reasonably considered independent and separate works in
124 | themselves, then this License, and its terms, do not apply to those
125 | sections when you distribute them as separate works. But when you
126 | distribute the same sections as part of a whole which is a work based
127 | on the Program, the distribution of the whole must be on the terms of
128 | this License, whose permissions for other licensees extend to the
129 | entire whole, and thus to each and every part regardless of who wrote
130 | it.
131 |
132 | Thus, it is not the intent of this section to claim rights or contest
133 | your rights to work written entirely by you; rather, the intent is to
134 | exercise the right to control the distribution of derivative or
135 | collective works based on the Program.
136 |
137 | In addition, mere aggregation of another work not based on the Program
138 | with the Program (or with a work based on the Program) on a volume of
139 | a storage or distribution medium does not bring the other work under
140 | the scope of this License.
141 |
142 | **3.** You may copy and distribute the Program (or a work based on it,
143 | under Section 2) in object code or executable form under the terms of
144 | Sections 1 and 2 above provided that you also do one of the following:
145 |
146 |
147 | **a)** Accompany it with the complete corresponding machine-readable
148 | source code, which must be distributed under the terms of Sections 1
149 | and 2 above on a medium customarily used for software interchange; or,
150 |
151 |
152 | **b)** Accompany it with a written offer, valid for at least three
153 | years, to give any third party, for a charge no more than your cost of
154 | physically performing source distribution, a complete machine-readable
155 | copy of the corresponding source code, to be distributed under the
156 | terms of Sections 1 and 2 above on a medium customarily used for
157 | software interchange; or,
158 |
159 |
160 | **c)** Accompany it with the information you received as to the offer
161 | to distribute corresponding source code. (This alternative is allowed
162 | only for noncommercial distribution and only if you received the
163 | program in object code or executable form with such an offer, in
164 | accord with Subsection b above.)
165 |
166 | The source code for a work means the preferred form of the work for
167 | making modifications to it. For an executable work, complete source
168 | code means all the source code for all modules it contains, plus any
169 | associated interface definition files, plus the scripts used to
170 | control compilation and installation of the executable. However, as a
171 | special exception, the source code distributed need not include
172 | anything that is normally distributed (in either source or binary
173 | form) with the major components (compiler, kernel, and so on) of the
174 | operating system on which the executable runs, unless that component
175 | itself accompanies the executable.
176 |
177 | If distribution of executable or object code is made by offering
178 | access to copy from a designated place, then offering equivalent
179 | access to copy the source code from the same place counts as
180 | distribution of the source code, even though third parties are not
181 | compelled to copy the source along with the object code.
182 |
183 | **4.** You may not copy, modify, sublicense, or distribute the Program
184 | except as expressly provided under this License. Any attempt otherwise
185 | to copy, modify, sublicense or distribute the Program is void, and
186 | will automatically terminate your rights under this License. However,
187 | parties who have received copies, or rights, from you under this
188 | License will not have their licenses terminated so long as such
189 | parties remain in full compliance.
190 |
191 | **5.** You are not required to accept this License, since you have not
192 | signed it. However, nothing else grants you permission to modify or
193 | distribute the Program or its derivative works. These actions are
194 | prohibited by law if you do not accept this License. Therefore, by
195 | modifying or distributing the Program (or any work based on the
196 | Program), you indicate your acceptance of this License to do so, and
197 | all its terms and conditions for copying, distributing or modifying
198 | the Program or works based on it.
199 |
200 | **6.** Each time you redistribute the Program (or any work based on
201 | the Program), the recipient automatically receives a license from the
202 | original licensor to copy, distribute or modify the Program subject to
203 | these terms and conditions. You may not impose any further
204 | restrictions on the recipients' exercise of the rights granted herein.
205 | You are not responsible for enforcing compliance by third parties to
206 | this License.
207 |
208 | **7.** If, as a consequence of a court judgment or allegation of
209 | patent infringement or for any other reason (not limited to patent
210 | issues), conditions are imposed on you (whether by court order,
211 | agreement or otherwise) that contradict the conditions of this
212 | License, they do not excuse you from the conditions of this License.
213 | If you cannot distribute so as to satisfy simultaneously your
214 | obligations under this License and any other pertinent obligations,
215 | then as a consequence you may not distribute the Program at all. For
216 | example, if a patent license would not permit royalty-free
217 | redistribution of the Program by all those who receive copies directly
218 | or indirectly through you, then the only way you could satisfy both it
219 | and this License would be to refrain entirely from distribution of the
220 | Program.
221 |
222 | If any portion of this section is held invalid or unenforceable under
223 | any particular circumstance, the balance of the section is intended to
224 | apply and the section as a whole is intended to apply in other
225 | circumstances.
226 |
227 | It is not the purpose of this section to induce you to infringe any
228 | patents or other property right claims or to contest validity of any
229 | such claims; this section has the sole purpose of protecting the
230 | integrity of the free software distribution system, which is
231 | implemented by public license practices. Many people have made
232 | generous contributions to the wide range of software distributed
233 | through that system in reliance on consistent application of that
234 | system; it is up to the author/donor to decide if he or she is willing
235 | to distribute software through any other system and a licensee cannot
236 | impose that choice.
237 |
238 | This section is intended to make thoroughly clear what is believed to
239 | be a consequence of the rest of this License.
240 |
241 | **8.** If the distribution and/or use of the Program is restricted in
242 | certain countries either by patents or by copyrighted interfaces, the
243 | original copyright holder who places the Program under this License
244 | may add an explicit geographical distribution limitation excluding
245 | those countries, so that distribution is permitted only in or among
246 | countries not thus excluded. In such case, this License incorporates
247 | the limitation as if written in the body of this License.
248 |
249 | **9.** The Free Software Foundation may publish revised and/or new
250 | versions of the General Public License from time to time. Such new
251 | versions will be similar in spirit to the present version, but may
252 | differ in detail to address new problems or concerns.
253 |
254 | Each version is given a distinguishing version number. If the Program
255 | specifies a version number of this License which applies to it and
256 | "any later version", you have the option of following the terms and
257 | conditions either of that version or of any later version published by
258 | the Free Software Foundation. If the Program does not specify a
259 | version number of this License, you may choose any version ever
260 | published by the Free Software Foundation.
261 |
262 | **10.** If you wish to incorporate parts of the Program into other
263 | free programs whose distribution conditions are different, write to
264 | the author to ask for permission. For software which is copyrighted by
265 | the Free Software Foundation, write to the Free Software Foundation;
266 | we sometimes make exceptions for this. Our decision will be guided by
267 | the two goals of preserving the free status of all derivatives of our
268 | free software and of promoting the sharing and reuse of software
269 | generally.
270 |
271 | **NO WARRANTY**
272 |
273 | **11.** BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO
274 | WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW.
275 | EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR
276 | OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY
277 | KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE
278 | IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
279 | PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE
280 | PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME
281 | THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
282 |
283 | **12.** IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN
284 | WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY
285 | AND/OR REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU
286 | FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR
287 | CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE
288 | PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING
289 | RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A
290 | FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF
291 | SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
292 | DAMAGES.
293 |
294 | ### END OF TERMS AND CONDITIONS
295 |
296 | ### How to Apply These Terms to Your New Programs
297 |
298 | If you develop a new program, and you want it to be of the greatest
299 | possible use to the public, the best way to achieve this is to make it
300 | free software which everyone can redistribute and change under these
301 | terms.
302 |
303 | To do so, attach the following notices to the program. It is safest to
304 | attach them to the start of each source file to most effectively
305 | convey the exclusion of warranty; and each file should have at least
306 | the "copyright" line and a pointer to where the full notice is found.
307 |
308 | one line to give the program's name and an idea of what it does.
309 | Copyright (C) yyyy name of author
310 |
311 | This program is free software; you can redistribute it and/or
312 | modify it under the terms of the GNU General Public License
313 | as published by the Free Software Foundation; either version 2
314 | of the License, or (at your option) any later version.
315 |
316 | This program is distributed in the hope that it will be useful,
317 | but WITHOUT ANY WARRANTY; without even the implied warranty of
318 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
319 | GNU General Public License for more details.
320 |
321 | You should have received a copy of the GNU General Public License
322 | along with this program; if not, write to the Free Software
323 | Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
324 |
325 | Also add information on how to contact you by electronic and paper
326 | mail.
327 |
328 | If the program is interactive, make it output a short notice like this
329 | when it starts in an interactive mode:
330 |
331 | Gnomovision version 69, Copyright (C) year name of author
332 | Gnomovision comes with ABSOLUTELY NO WARRANTY; for details
333 | type `show w'. This is free software, and you are welcome
334 | to redistribute it under certain conditions; type `show c'
335 | for details.
336 |
337 | The hypothetical commands \`show w' and \`show c' should show the
338 | appropriate parts of the General Public License. Of course, the
339 | commands you use may be called something other than \`show w' and
340 | \`show c'; they could even be mouse-clicks or menu items--whatever
341 | suits your program.
342 |
343 | You should also get your employer (if you work as a programmer) or
344 | your school, if any, to sign a "copyright disclaimer" for the program,
345 | if necessary. Here is a sample; alter the names:
346 |
347 | Yoyodyne, Inc., hereby disclaims all copyright
348 | interest in the program `Gnomovision'
349 | (which makes passes at compilers) written
350 | by James Hacker.
351 |
352 | signature of Ty Coon, 1 April 1989
353 | Ty Coon, President of Vice
354 |
355 | This General Public License does not permit incorporating your program
356 | into proprietary programs. If your program is a subroutine library,
357 | you may consider it more useful to permit linking proprietary
358 | applications with the library. If this is what you want to do, use the
359 | [GNU Lesser General Public
360 | License](https://www.gnu.org/licenses/lgpl.html) instead of this
361 | License.
362 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | [](https://travis-ci.org/felixarntz/wp-psr-cache)
2 | [](#contributors)
3 | [](https://codeclimate.com/github/felixarntz/wp-psr-cache)
4 | [](https://codeclimate.com/github/felixarntz/wp-psr-cache/coverage)
5 | [](https://packagist.org/packages/felixarntz/wp-psr-cache)
6 | [](https://packagist.org/packages/felixarntz/wp-psr-cache)
7 |
8 | # WP PSR Cache
9 |
10 | Object cache implementation for WordPress that acts as an adapter for PSR-6 and PSR-16 caching libraries.
11 |
12 | ## What do PSR-6 and PSR-16 mean?
13 |
14 | [PSR-6](http://www.php-fig.org/psr/psr-6/) and [PSR-16](http://www.php-fig.org/psr/psr-16/) are standards established by the [PHP-FIG](http://www.php-fig.org/) organization. These standards are commonly used in PHP projects of any kind (WordPress is unfortunately an exception), and since this library acts as an adapter, you can use any compatible caching library of your choice with WordPress now. Popular examples include the [Symfony Cache Component](https://github.com/symfony/cache) or [Stash](https://github.com/tedious/Stash).
15 |
16 | ## Features
17 |
18 | * Any PSR-6 or PSR-16 cache implementation can be used
19 | * Persistent and non-persistent cache implementations can be individually specified
20 | * Support for reading/writing/deleting multiple cache keys at once
21 | * Only checks persistent cache if value not already present in non-persistent cache
22 | * Full multisite support, including site and network switching
23 | * Allows registration of further cache implementations for fine-grained control per cache group
24 |
25 | ## How to Install
26 |
27 | Require this library as a dependency when managing your project with Composer (for example by using [Bedrock](https://github.com/roots/bedrock)). You also have to install an actual [PSR-6](https://packagist.org/providers/psr/cache-implementation) or [PSR-16](https://packagist.org/providers/psr/simple-cache-implementation) cache implementation.
28 |
29 | After the installation, you need to move the `includes/object-cache.php` file into your `wp-content` directory. If you prefer, you can also automate that process by adding the following to your project's `composer.json`:
30 |
31 | ```
32 | "scripts": {
33 | "post-install-cmd": [
34 | "cp -rp web/app/mu-plugins/wp-psr-cache/includes/object-cache.php web/app/object-cache.php"
35 | ]
36 | }
37 | ```
38 |
39 | Then, replace the inline comment in the `object-cache.php` file with the actual instantiations of the classes you want to use. You need to provide two implementations, one for the persistent cache and another for the non-persistent cache.
40 |
41 | To prevent conflicts with multiple WordPress installations accessing the same cache service, it is recommended to define a unique `WP_CACHE_KEY_PREFIX` constant in your `wp-config.php` file.
42 |
43 | ### Example
44 |
45 | The following example uses the `symfony/cache` library, so you have to require it in your `composer.json`. It then uses that library's Memcached implementation as persistent cache and its array storage as non-persistent cache.
46 |
47 | ```php
48 | addServer( '127.0.0.1', 11211, 20 );
73 |
74 | wp_cache_start( new MemcachedCache( $memcached ), new ArrayCache() );
75 | }
76 |
77 | wp_psr_start_cache();
78 |
79 | ```
80 |
81 | If you prefer to have more granular control and use more than just one persistent and one non-persistent cache, you can register additional cache adapters using the [`LeavesAndLove\WpPsrCache\CacheSelector\CacheSelector`](https://github.com/felixarntz/wp-psr-cache/blob/master/src/CacheSelector/CacheSelector.php) interface. The implementation used by the object cache can easily be retrieved via `wp_object_cache()->getSelector()`.
82 |
83 | ## Requirements
84 |
85 | * PHP >= 7.0
86 |
87 | ## Contributors
88 |
89 | Thanks goes to these wonderful people ([emoji key](https://github.com/kentcdodds/all-contributors#emoji-key)):
90 |
91 |
92 |
93 | | [
Felix Arntz](https://leaves-and-love.net)
[💻](https://github.com/felixarntz/wp-psr-cache/commits?author=felixarntz "Code") [🐛](https://github.com/felixarntz/wp-psr-cache/issues?q=author%3Afelixarntz "Bug reports") [📖](https://github.com/felixarntz/wp-psr-cache/commits?author=felixarntz "Documentation") [💡](#example-felixarntz "Examples") [🤔](#ideas-felixarntz "Ideas, Planning, & Feedback") [⚠️](https://github.com/felixarntz/wp-psr-cache/commits?author=felixarntz "Tests") | [
Alain Schlesser](https://www.alainschlesser.com/)
[💻](https://github.com/felixarntz/wp-psr-cache/commits?author=schlessera "Code") [🐛](https://github.com/felixarntz/wp-psr-cache/issues?q=author%3Aschlessera "Bug reports") [👀](#review-schlessera "Reviewed Pull Requests") | [
Thorsten Frommen](https://tfrommen.de)
[👀](#review-tfrommen "Reviewed Pull Requests") | [
Jip](http://www.jipmoors.nl)
[🤔](#ideas-moorscode "Ideas, Planning, & Feedback") |
94 | | :---: | :---: | :---: | :---: |
95 |
96 |
97 | This project follows the [all-contributors](https://github.com/kentcdodds/all-contributors) specification. Contributions of any kind welcome!
--------------------------------------------------------------------------------
/composer.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "felixarntz/wp-psr-cache",
3 | "description": "Object cache implementation for WordPress that acts as an adapter for PSR-6 and PSR-16 caching libraries.",
4 | "version": "1.0.0",
5 | "type": "wordpress-muplugin",
6 | "license": "GPL-2.0-or-later",
7 | "keywords": [
8 | "wordpress",
9 | "mu-plugin",
10 | "cache",
11 | "psr-6",
12 | "psr-16",
13 | "adapter"
14 | ],
15 | "homepage": "https://github.com/felixarntz/wp-psr-cache",
16 | "authors": [
17 | {
18 | "name": "Felix Arntz",
19 | "email": "felix-arntz@leaves-and-love.net",
20 | "homepage": "https://leaves-and-love.net",
21 | "role": "Developer"
22 | }
23 | ],
24 | "support": {
25 | "issues": "https://github.com/felixarntz/wp-psr-cache/issues"
26 | },
27 | "autoload": {
28 | "psr-4": {
29 | "LeavesAndLove\\WpPsrCache\\": "src"
30 | }
31 | },
32 | "autoload-dev": {
33 | "psr-4": {
34 | "LeavesAndLove\\WpPsrCache\\Tests\\": "tests"
35 | }
36 | },
37 | "require": {
38 | "php": "^7.0",
39 | "composer/installers": "~1.0",
40 | "psr/cache": "*",
41 | "psr/simple-cache": "*"
42 | },
43 | "require-dev": {
44 | "phpunit/phpunit": "6.2.*",
45 | "squizlabs/php_codesniffer": "3.*"
46 | }
47 | }
48 |
--------------------------------------------------------------------------------
/includes/functions.php:
--------------------------------------------------------------------------------
1 | create( $persistent_cache );
44 | $non_persistent_cache_adapter = $adapter_factory->create( $non_persistent_cache );
45 |
46 | $selector = $selector_factory->create( $persistent_cache_adapter, $non_persistent_cache_adapter );
47 | $cache = $cache_factory->create( $selector );
48 |
49 | ObjectCacheService::setInstance( $cache );
50 | }
51 |
52 | /**
53 | * Adds a group or list of groups to the global cache groups.
54 | *
55 | * @since 1.0.0
56 | * @see WpCacheKeyGen::addGlobalGroups()
57 | *
58 | * @param string|array $groups A group or an array of groups to add.
59 | */
60 | function wp_cache_add_global_groups( $groups ) {
61 | wp_object_cache()->getKeygen()->addGlobalGroups( (array) $groups );
62 | }
63 |
64 | /**
65 | * Adds a group or list of groups to the network cache groups.
66 | *
67 | * @since 1.0.0
68 | * @see WpCacheKeyGen::addNetworkGroups()
69 | *
70 | * @param string|array $groups A group or an array of groups to add.
71 | */
72 | function wp_cache_add_network_groups( $groups ) {
73 | wp_object_cache()->getKeygen()->addNetworkGroups( (array) $groups );
74 | }
75 |
76 | /**
77 | * Adds a group or list of groups to the non-persistent cache groups.
78 | *
79 | * @since 1.0.0
80 | * @see CacheSelector::addNonPersistentGroups()
81 | *
82 | * @param string|array $groups A group or an array of groups to add.
83 | */
84 | function wp_cache_add_non_persistent_groups( $groups ) {
85 | wp_object_cache()->getSelector()->addNonPersistentGroups( (array) $groups );
86 | }
87 |
88 | /**
89 | * Switches the internal site ID.
90 | *
91 | * @since 1.0.0
92 | * @see WpCacheKeyGen::switchSiteContext()
93 | *
94 | * @param int $site_id Site ID.
95 | */
96 | function wp_cache_switch_to_site( $site_id ) {
97 | wp_object_cache()->getKeygen()->switchSiteContext( (int) $site_id );
98 | }
99 |
100 | /**
101 | * Switches the internal network ID.
102 | *
103 | * @since 1.0.0
104 | * @see WpCacheKeyGen::switchNetworkContext()
105 | *
106 | * @param int $network_id Network ID.
107 | */
108 | function wp_cache_switch_to_network( $network_id ) {
109 | wp_object_cache()->getKeygen()->switchNetworkContext( (int) $network_id );
110 | }
111 |
112 | /**
113 | * Obtains a value from the cache.
114 | *
115 | * @since 1.0.0
116 | * @see ObjectCache::get()
117 | *
118 | * @param string $key The key of this item in the cache.
119 | * @param string $group Optional. The group of this item in the cache. Default empty string.
120 | * @param bool $force Optional. Whether to force an update of the non-persistent cache
121 | * from the persistent cache. Default false.
122 | * @param bool &$found Optional. Whether the key was found in the cache (passed by reference).
123 | * Disambiguates a return of false, a storable value. Default null.
124 | * @return mixed The value of the item from the cache, or false in case of cache miss.
125 | */
126 | function wp_cache_get( $key, $group = '', $force = false, &$found = null ) {
127 | $found = (bool) $found;
128 |
129 | return wp_object_cache()->get( $key, $group, $force, $found );
130 | }
131 |
132 | /**
133 | * Stores a value in the cache.
134 | *
135 | * @since 1.0.0
136 | * @see ObjectCache::set()
137 | *
138 | * @param string $key The key of the item to store.
139 | * @param mixed $value The value of the item to store. Must be serializable.
140 | * @param string $group Optional. The group of the item to store. Default empty string.
141 | * @param int $expiration Optional. When to expire the value, passed in seconds. Default 0 (no expiration).
142 | * @return bool True on success, false on failure.
143 | */
144 | function wp_cache_set( $key, $value, $group = '', $expiration = 0 ) {
145 | return wp_object_cache()->set( $key, $value, $group, $expiration );
146 | }
147 |
148 | /**
149 | * Stores a value in the cache if its key is not already set.
150 | *
151 | * @since 1.0.0
152 | * @see ObjectCache::add()
153 | *
154 | * @param string $key The key of the item to store.
155 | * @param mixed $value The value of the item to store. Must be serializable.
156 | * @param string $group Optional. The group of the item to store. Default empty string.
157 | * @param int $expiration Optional. When to expire the value, passed in seconds. Default 0 (no expiration).
158 | * @return bool True on success, false on failure.
159 | */
160 | function wp_cache_add( $key, $value, $group = '', $expiration = 0 ) {
161 | if ( wp_suspend_cache_addition() ) {
162 | return false;
163 | }
164 |
165 | return wp_object_cache()->add( $key, $value, $group, $expiration );
166 | }
167 |
168 | /**
169 | * Stores a value in the cache if its key is already set.
170 | *
171 | * @since 1.0.0
172 | * @see ObjectCache::replace()
173 | *
174 | * @param string $key The key of the item to store.
175 | * @param mixed $value The value of the item to store. Must be serializable.
176 | * @param string $group Optional. The group of the item to store. Default empty string.
177 | * @param int $expiration Optional. When to expire the value, passed in seconds. Default 0 (no expiration).
178 | * @return bool True on success, false on failure.
179 | */
180 | function wp_cache_replace( $key, $value, $group = '', $expiration = 0 ) {
181 | return wp_object_cache()->replace( $key, $value, $group, $expiration );
182 | }
183 |
184 | /**
185 | * Increments a numeric value in the cache.
186 | *
187 | * @since 1.0.0
188 | * @see ObjectCache::increment()
189 | *
190 | * @param string $key The key of the item to increment its value.
191 | * @param int $offset Optional. The amount by which to increment the value. Default 1.
192 | * @param string $group Optional. The group of the item to increment. Default empty string.
193 | * @return int|bool The item's new value on success, false on failure.
194 | */
195 | function wp_cache_incr( $key, $offset = 1, $group = '' ) {
196 | return wp_object_cache()->increment( $key, $offset, $group );
197 | }
198 |
199 | /**
200 | * Decrements a numeric value in the cache.
201 | *
202 | * @since 1.0.0
203 | * @see ObjectCache::decrement()
204 | *
205 | * @param string $key The key of the item to decrement its value.
206 | * @param int $offset Optional. The amount by which to decrement the value. Default 1.
207 | * @param string $group Optional. The group of the item to decrement. Default empty string.
208 | * @return int|bool The item's new value on success, false on failure.
209 | */
210 | function wp_cache_decr( $key, $offset = 1, $group = '' ) {
211 | return wp_object_cache()->decrement( $key, $offset, $group );
212 | }
213 |
214 | /**
215 | * Deletes a value from the cache.
216 | *
217 | * @since 1.0.0
218 | * @see ObjectCache::delete()
219 | *
220 | * @param string $key The key of the item to delete.
221 | * @param string $group Optional. The group of the item to delete. Default empty string.
222 | * @return bool True on success, false on failure.
223 | */
224 | function wp_cache_delete( $key, $group = '' ) {
225 | return wp_object_cache()->delete( $key, $group );
226 | }
227 |
228 | /**
229 | * Deletes all values from the cache.
230 | *
231 | * @since 1.0.0
232 | * @see ObjectCache::flush()
233 | *
234 | * @return bool True on success, false on failure.
235 | */
236 | function wp_cache_flush() {
237 | return wp_object_cache()->flush();
238 | }
239 |
240 | /**
241 | * Initializes the object cache.
242 | *
243 | * @since 1.0.0
244 | */
245 | function wp_cache_init() {
246 | // This ensures an exception is thrown if no object cache has been set before this point.
247 | ObjectCacheService::getInstance();
248 | }
249 |
250 | /**
251 | * Closes the cache.
252 | *
253 | * @since 1.0.0
254 | *
255 | * @return bool True on success, false on failure.
256 | */
257 | function wp_cache_close() {
258 | return true;
259 | }
260 |
261 | /**
262 | * Determines whether a value is present in the cache.
263 | *
264 | * @since 1.0.0
265 | * @see ObjectCache::has()
266 | *
267 | * @param string $key The key of the item in the cache.
268 | * @param string $group Optional. The group of the item in the cache. Default empty string.
269 | * @return bool True if the value is present, false otherwise.
270 | */
271 | function wp_cache_has( $key, $group = '' ) {
272 | return wp_object_cache()->has( $key, $group );
273 | }
274 |
275 | /**
276 | * Obtains multiple values from the cache.
277 | *
278 | * @since 1.0.0
279 | * @see ObjectCache::getMultiple()
280 | *
281 | * @param array $keys The list of keys for the items in the cache.
282 | * @param string $group Optional. The group of the items in the cache. Default empty string.
283 | * @param bool $force Optional. Whether to force an update of the non-persistent cache
284 | * from the persistent cache. Default false.
285 | * @return array List of key => value pairs. For cache misses, false will be used as value.
286 | */
287 | function wp_cache_get_multi( $keys, $group = '', $force = false ) {
288 | return wp_object_cache()->getMultiple( $keys, $group, $force );
289 | }
290 |
291 | /**
292 | * Stores multiple values in the cache.
293 | *
294 | * @since 1.0.0
295 | * @see ObjectCache::setMultiple()
296 | *
297 | * @param array $keys The list of key => value pairs to store.
298 | * @param string $group Optional. The group of the items to store. Default empty string.
299 | * @param int $expiration Optional. When to expire the value, passed in seconds. Default 0 (no expiration).
300 | * @return bool True on success, false on failure.
301 | */
302 | function wp_cache_set_multi( $values, $group = '', $expiration = 0 ) {
303 | return wp_object_cache()->setMultiple( $values, $group, $expiration );
304 | }
305 |
306 | /**
307 | * Deletes multiple values from the cache.
308 | *
309 | * @since 1.0.0
310 | * @see ObjectCache::deleteMultiple()
311 | *
312 | * @param array $keys The list of keys for the items in the cache to delete.
313 | * @param string $group Optional. The group of the items to delete. Default empty string.
314 | * @return bool True on success, false on failure.
315 | */
316 | function wp_cache_delete_multi( $keys, $group = '' ) {
317 | return wp_object_cache()->deleteMultiple( $keys, $group );
318 | }
319 |
320 | /**
321 | * Builds the full internal cache key for a given key and group.
322 | *
323 | * @since 1.0.0
324 | * @see WpCacheKeyGen::generate()
325 | *
326 | * @param string $key A cache key.
327 | * @param string $group A cache group.
328 | * @return string The full cache key to use with cache implementations.
329 | */
330 | function wp_cache_get_key( $key, $group = '' ) {
331 | return wp_object_cache()->getKeygen()->generate( $key, $group );
332 | }
333 |
334 | /**
335 | * Gets the main object cache instance.
336 | *
337 | * @since 1.0.0
338 | *
339 | * @global WP_Object_Cache $wp_object_cache Object cache global instance.
340 | *
341 | * @return ObjectCache Main object cache instance.
342 | */
343 | function wp_object_cache() {
344 | return $GLOBALS['wp_object_cache'];
345 | }
346 |
347 | /**
348 | * Switches the internal site ID.
349 | *
350 | * This function exists for compatibility, but is actually incorrectly named.
351 | * It will not trigger a deprecated notice, but it's still outdated.
352 | *
353 | * @since 1.0.0
354 | * @deprecated 1.0.0 Use wp_cache_switch_to_site()
355 | * @see wp_cache_switch_to_site()
356 | *
357 | * @param int $blog_id Site ID.
358 | */
359 | function wp_cache_switch_to_blog( $blog_id ) {
360 | wp_cache_switch_to_site( $blog_id );
361 |
362 | // When `wp_cache_switch_to_blog()` is called right after multisite initialization, it must set the network.
363 | $keygen = wp_object_cache()->getKeygen();
364 | if ( 0 === $keygen->getNetworkContext() ) {
365 | if ( ! empty( $GLOBALS['current_blog'] ) && (int) $blog_id === (int) $GLOBALS['current_blog']->id ) {
366 | $site = $GLOBALS['current_blog'];
367 | } else {
368 | $site = get_site( $blog_id );
369 | }
370 |
371 | $keygen->switchNetworkContext( $site->network_id );
372 | }
373 | }
374 |
--------------------------------------------------------------------------------
/includes/object-cache.php:
--------------------------------------------------------------------------------
1 | (https://leaves-and-love.net)",
17 | "devDependencies": {
18 | "all-contributors-cli": "^4.10.1"
19 | }
20 | }
21 |
--------------------------------------------------------------------------------
/phpcs.xml:
--------------------------------------------------------------------------------
1 |
2 |
';
518 | $out[] = 'Cache Hits: ' . $this->cacheHits . '
';
519 | $out[] = 'Cache Misses: ' . $this->cacheMisses . '
';
520 | $out[] = 'Persistent Cache Client: ' . get_class( $this->selector->selectPersistentCache( '' )->getClient() ) . '
';
521 | $out[] = 'Non-Persistent Cache Client: ' . get_class( $this->selector->selectNonPersistentCache( '' )->getClient() ) . '
';
522 | $out[] = '