Priority File Manager

πŸ“ public_html
Base Directory:
/home/toolrmtm/crazyespresso.com/wp-includes
NameTypeSizeActions
πŸ“ .. Folder -
πŸ“ ID3 Folder -
πŸ“ IXR Folder -
πŸ“ PHPMailer Folder -
πŸ“ Requests Folder -
πŸ“ SimplePie Folder -
πŸ“ Text Folder -
πŸ“ abilities-api Folder -
πŸ“„ abilities-api.php File 24369
Edit Download
πŸ“„ abilities.php File 7983
Edit Download
πŸ“„ admin-bar.php File 36966
Edit Download
πŸ“ assets Folder -
πŸ“„ atomlib.php File 12181
Edit Download
πŸ“„ author-template.php File 19391
Edit Download
πŸ“ block-bindings Folder -
πŸ“„ block-bindings.php File 7526
Edit Download
πŸ“„ block-editor.php File 29282
Edit Download
πŸ“„ block-i18n.json File 316
Edit Download
πŸ“ block-patterns Folder -
πŸ“„ block-patterns.php File 13213
Edit Download
πŸ“ block-supports Folder -
πŸ“„ block-template-utils.php File 62484
Edit Download
πŸ“„ block-template.php File 15359
Edit Download
πŸ“ blocks Folder -
πŸ“„ blocks.php File 114739
Edit Download
πŸ“„ bookmark-template.php File 12768
Edit Download
πŸ“„ bookmark.php File 15427
Edit Download
πŸ“„ cache-compat.php File 10078
Edit Download
πŸ“„ cache.php File 13486
Edit Download
πŸ“„ canonical.php File 34645
Edit Download
πŸ“„ capabilities.php File 43652
Edit Download
πŸ“„ category-template.php File 57045
Edit Download
πŸ“„ category.php File 12829
Edit Download
πŸ“ certificates Folder -
πŸ“„ class-IXR.php File 2616
Edit Download
πŸ“„ class-avif-info.php File 29615
Edit Download
πŸ“„ class-feed.php File 539
Edit Download
πŸ“„ class-http.php File 367
Edit Download
πŸ“„ class-json.php File 43676
Edit Download
πŸ“„ class-oembed.php File 401
Edit Download
πŸ“„ class-phpass.php File 6771
Edit Download
πŸ“„ class-phpmailer.php File 664
Edit Download
πŸ“„ class-pop3.php File 21121
Edit Download
πŸ“„ class-requests.php File 2237
Edit Download
πŸ“„ class-simplepie.php File 453
Edit Download
πŸ“„ class-smtp.php File 457
Edit Download
πŸ“„ class-snoopy.php File 37715
Edit Download
πŸ“„ class-walker-category-dropdown.php File 2469
Edit Download
πŸ“„ class-walker-category.php File 8477
Edit Download
πŸ“„ class-walker-comment.php File 14221
Edit Download
πŸ“„ class-walker-nav-menu.php File 12044
Edit Download
πŸ“„ class-walker-page-dropdown.php File 2710
Edit Download
πŸ“„ class-walker-page.php File 7612
Edit Download
πŸ“„ class-wp-admin-bar.php File 17874
Edit Download
πŸ“„ class-wp-ajax-response.php File 5266
Edit Download
πŸ“„ class-wp-application-passwords.php File 17099
Edit Download
πŸ“„ class-wp-block-bindings-registry.php File 8482
Edit Download
πŸ“„ class-wp-block-bindings-source.php File 2992
Edit Download
πŸ“„ class-wp-block-editor-context.php File 1350
Edit Download
πŸ“„ class-wp-block-list.php File 4713
Edit Download
πŸ“„ class-wp-block-metadata-registry.php File 11895
Edit Download
πŸ“„ class-wp-block-parser-block.php File 2555
Edit Download
πŸ“„ class-wp-block-parser-frame.php File 2017
Edit Download
πŸ“„ class-wp-block-parser.php File 11516
Edit Download
πŸ“„ class-wp-block-pattern-categories-registry.php File 5450
Edit Download
πŸ“„ class-wp-block-patterns-registry.php File 10851
Edit Download
πŸ“„ class-wp-block-processor.php File 69469
Edit Download
πŸ“„ class-wp-block-styles-registry.php File 6497
Edit Download
πŸ“„ class-wp-block-supports.php File 5626
Edit Download
πŸ“„ class-wp-block-template.php File 2033
Edit Download
πŸ“„ class-wp-block-templates-registry.php File 7193
Edit Download
πŸ“„ class-wp-block-type-registry.php File 5030
Edit Download
πŸ“„ class-wp-block-type.php File 17265
Edit Download
πŸ“„ class-wp-block.php File 24812
Edit Download
πŸ“„ class-wp-classic-to-block-menu-converter.php File 4070
Edit Download
πŸ“„ class-wp-comment-query.php File 48804
Edit Download
πŸ“„ class-wp-comment.php File 9437
Edit Download
πŸ“„ class-wp-customize-control.php File 26119
Edit Download
πŸ“„ class-wp-customize-manager.php File 203139
Edit Download
πŸ“„ class-wp-customize-nav-menus.php File 58013
Edit Download
πŸ“„ class-wp-customize-panel.php File 10710
Edit Download
πŸ“„ class-wp-customize-section.php File 11209
Edit Download
πŸ“„ class-wp-customize-setting.php File 29962
Edit Download
πŸ“„ class-wp-customize-widgets.php File 72607
Edit Download
πŸ“„ class-wp-date-query.php File 36147
Edit Download
πŸ“„ class-wp-dependencies.php File 15381
Edit Download
πŸ“„ class-wp-dependency.php File 2633
Edit Download
πŸ“„ class-wp-duotone.php File 40783
Edit Download
πŸ“„ class-wp-editor.php File 72335
Edit Download
πŸ“„ class-wp-embed.php File 15931
Edit Download
πŸ“„ class-wp-error.php File 7502
Edit Download
πŸ“„ class-wp-exception.php File 253
Edit Download
πŸ“„ class-wp-fatal-error-handler.php File 8150
Edit Download
πŸ“„ class-wp-feed-cache-transient.php File 3304
Edit Download
πŸ“„ class-wp-feed-cache.php File 969
Edit Download
πŸ“„ class-wp-hook.php File 16674
Edit Download
πŸ“„ class-wp-http-cookie.php File 7389
Edit Download
πŸ“„ class-wp-http-curl.php File 13261
Edit Download
πŸ“„ class-wp-http-encoding.php File 6689
Edit Download
πŸ“„ class-wp-http-ixr-client.php File 3501
Edit Download
πŸ“„ class-wp-http-proxy.php File 5980
Edit Download
πŸ“„ class-wp-http-requests-hooks.php File 2022
Edit Download
πŸ“„ class-wp-http-requests-response.php File 4400
Edit Download
πŸ“„ class-wp-http-response.php File 2977
Edit Download
πŸ“„ class-wp-http-streams.php File 16859
Edit Download
πŸ“„ class-wp-http.php File 41570
Edit Download
πŸ“„ class-wp-image-editor-gd.php File 20705
Edit Download
πŸ“„ class-wp-image-editor-imagick.php File 36977
Edit Download
πŸ“„ class-wp-image-editor.php File 17415
Edit Download
πŸ“„ class-wp-list-util.php File 7443
Edit Download
πŸ“„ class-wp-locale-switcher.php File 6776
Edit Download
πŸ“„ class-wp-locale.php File 16883
Edit Download
πŸ“„ class-wp-matchesmapregex.php File 1828
Edit Download
πŸ“„ class-wp-meta-query.php File 30533
Edit Download
πŸ“„ class-wp-metadata-lazyloader.php File 6833
Edit Download
πŸ“„ class-wp-navigation-fallback.php File 9193
Edit Download
πŸ“„ class-wp-network-query.php File 19887
Edit Download
πŸ“„ class-wp-network.php File 12296
Edit Download
πŸ“„ class-wp-object-cache.php File 17524
Edit Download
πŸ“„ class-wp-oembed-controller.php File 6905
Edit Download
πŸ“„ class-wp-oembed.php File 31670
Edit Download
πŸ“„ class-wp-paused-extensions-storage.php File 5111
Edit Download
πŸ“„ class-wp-phpmailer.php File 4348
Edit Download
πŸ“„ class-wp-plugin-dependencies.php File 25315
Edit Download
πŸ“„ class-wp-post-type.php File 30680
Edit Download
πŸ“„ class-wp-post.php File 6491
Edit Download
πŸ“„ class-wp-query.php File 163744
Edit Download
πŸ“„ class-wp-recovery-mode-cookie-service.php File 6877
Edit Download
πŸ“„ class-wp-recovery-mode-email-service.php File 11183
Edit Download
πŸ“„ class-wp-recovery-mode-key-service.php File 4884
Edit Download
πŸ“„ class-wp-recovery-mode-link-service.php File 3463
Edit Download
πŸ“„ class-wp-recovery-mode.php File 11453
Edit Download
πŸ“„ class-wp-rewrite.php File 63687
Edit Download
πŸ“„ class-wp-role.php File 2523
Edit Download
πŸ“„ class-wp-roles.php File 9394
Edit Download
πŸ“„ class-wp-script-modules.php File 31882
Edit Download
πŸ“„ class-wp-scripts.php File 34177
Edit Download
πŸ“„ class-wp-session-tokens.php File 7319
Edit Download
πŸ“„ class-wp-simplepie-file.php File 3552
Edit Download
πŸ“„ class-wp-simplepie-sanitize-kses.php File 1910
Edit Download
πŸ“„ class-wp-site-query.php File 31655
Edit Download
πŸ“„ class-wp-site.php File 7467
Edit Download
πŸ“„ class-wp-speculation-rules.php File 7527
Edit Download
πŸ“„ class-wp-styles.php File 12144
Edit Download
πŸ“„ class-wp-tax-query.php File 19577
Edit Download
πŸ“„ class-wp-taxonomy.php File 18559
Edit Download
πŸ“„ class-wp-term-query.php File 40953
Edit Download
πŸ“„ class-wp-term.php File 5298
Edit Download
πŸ“„ class-wp-text-diff-renderer-inline.php File 979
Edit Download
πŸ“„ class-wp-text-diff-renderer-table.php File 18880
Edit Download
πŸ“„ class-wp-textdomain-registry.php File 10481
Edit Download
πŸ“„ class-wp-theme-json-data.php File 1809
Edit Download
πŸ“„ class-wp-theme-json-resolver.php File 35738
Edit Download
πŸ“„ class-wp-theme-json-schema.php File 7367
Edit Download
πŸ“„ class-wp-theme-json.php File 164347
Edit Download
πŸ“„ class-wp-theme.php File 65810
Edit Download
πŸ“„ class-wp-token-map.php File 28618
Edit Download
πŸ“„ class-wp-url-pattern-prefixer.php File 4802
Edit Download
πŸ“„ class-wp-user-meta-session-tokens.php File 3011
Edit Download
πŸ“„ class-wp-user-query.php File 44166
Edit Download
πŸ“„ class-wp-user-request.php File 2305
Edit Download
πŸ“„ class-wp-user.php File 23044
Edit Download
πŸ“„ class-wp-walker.php File 13322
Edit Download
πŸ“„ class-wp-widget-factory.php File 3347
Edit Download
πŸ“„ class-wp-widget.php File 18429
Edit Download
πŸ“„ class-wp-xmlrpc-server.php File 215447
Edit Download
πŸ“„ class-wp.php File 26481
Edit Download
πŸ“„ class-wpdb.php File 118627
Edit Download
πŸ“„ class.wp-dependencies.php File 373
Edit Download
πŸ“„ class.wp-scripts.php File 343
Edit Download
πŸ“„ class.wp-styles.php File 338
Edit Download
πŸ“„ comment-template.php File 103145
Edit Download
πŸ“„ comment.php File 134069
Edit Download
πŸ“„ compat-utf8.php File 19554
Edit Download
πŸ“„ compat.php File 17830
Edit Download
πŸ“„ cron.php File 42988
Edit Download
πŸ“ css Folder -
πŸ“ customize Folder -
πŸ“„ date.php File 400
Edit Download
πŸ“„ default-constants.php File 11365
Edit Download
πŸ“„ default-filters.php File 37910
Edit Download
πŸ“„ default-widgets.php File 2295
Edit Download
πŸ“„ deprecated.php File 192644
Edit Download
πŸ“„ embed-template.php File 338
Edit Download
πŸ“„ embed.php File 38911
Edit Download
πŸ“„ error-protection.php File 4121
Edit Download
πŸ“„ feed-atom-comments.php File 5504
Edit Download
πŸ“„ feed-atom.php File 3121
Edit Download
πŸ“„ feed-rdf.php File 2668
Edit Download
πŸ“„ feed-rss.php File 1189
Edit Download
πŸ“„ feed-rss2-comments.php File 4136
Edit Download
πŸ“„ feed-rss2.php File 3799
Edit Download
πŸ“„ feed.php File 23579
Edit Download
πŸ“ fonts Folder -
πŸ“„ fonts.php File 9790
Edit Download
πŸ“„ formatting.php File 354741
Edit Download
πŸ“„ functions.php File 288600
Edit Download
πŸ“„ functions.wp-scripts.php File 15311
Edit Download
πŸ“„ functions.wp-styles.php File 8641
Edit Download
πŸ“„ general-template.php File 173004
Edit Download
πŸ“„ global-styles-and-settings.php File 21204
Edit Download
πŸ“ html-api Folder -
πŸ“„ http.php File 25878
Edit Download
πŸ“„ https-detection.php File 5857
Edit Download
πŸ“„ https-migration.php File 4741
Edit Download
πŸ“ images Folder -
πŸ“ interactivity-api Folder -
πŸ“ js Folder -
πŸ“„ kses.php File 83677
Edit Download
πŸ“ l10n Folder -
πŸ“„ l10n.php File 68797
Edit Download
πŸ“„ link-template.php File 160117
Edit Download
πŸ“„ load.php File 56510
Edit Download
πŸ“„ locale.php File 162
Edit Download
πŸ“„ media-template.php File 63197
Edit Download
πŸ“„ media.php File 221186
Edit Download
πŸ“„ meta.php File 66556
Edit Download
πŸ“„ ms-blogs.php File 25845
Edit Download
πŸ“„ ms-default-constants.php File 4921
Edit Download
πŸ“„ ms-default-filters.php File 6636
Edit Download
πŸ“„ ms-deprecated.php File 21759
Edit Download
πŸ“„ ms-files.php File 2857
Edit Download
πŸ“„ ms-functions.php File 91842
Edit Download
πŸ“„ ms-load.php File 19887
Edit Download
πŸ“„ ms-network.php File 3782
Edit Download
πŸ“„ ms-settings.php File 4204
Edit Download
πŸ“„ ms-site.php File 41717
Edit Download
πŸ“„ nav-menu-template.php File 25990
Edit Download
πŸ“„ nav-menu.php File 44373
Edit Download
πŸ“„ option.php File 105035
Edit Download
πŸ“ php-compat Folder -
πŸ“„ pluggable-deprecated.php File 6324
Edit Download
πŸ“„ pluggable.php File 127440
Edit Download
πŸ“„ plugin.php File 36501
Edit Download
πŸ“ pomo Folder -
πŸ“„ post-formats.php File 7102
Edit Download
πŸ“„ post-template.php File 68648
Edit Download
πŸ“„ post-thumbnail-template.php File 10879
Edit Download
πŸ“„ post.php File 296072
Edit Download
πŸ“„ query.php File 37095
Edit Download
πŸ“„ registration-functions.php File 200
Edit Download
πŸ“„ registration.php File 200
Edit Download
πŸ“ rest-api Folder -
πŸ“„ rest-api.php File 100654
Edit Download
πŸ“„ revision.php File 30741
Edit Download
πŸ“„ rewrite.php File 19490
Edit Download
πŸ“„ robots-template.php File 5185
Edit Download
πŸ“„ rss-functions.php File 255
Edit Download
πŸ“„ rss.php File 23203
Edit Download
πŸ“„ script-loader.php File 153993
Edit Download
πŸ“„ script-modules.php File 9911
Edit Download
πŸ“„ session.php File 258
Edit Download
πŸ“„ shortcodes.php File 24050
Edit Download
πŸ“ sitemaps Folder -
πŸ“„ sitemaps.php File 3238
Edit Download
πŸ“ sodium_compat Folder -
πŸ“„ speculative-loading.php File 8600
Edit Download
πŸ“„ spl-autoload-compat.php File 441
Edit Download
πŸ“ style-engine Folder -
πŸ“„ style-engine.php File 7563
Edit Download
πŸ“„ taxonomy.php File 177058
Edit Download
πŸ“„ template-canvas.php File 544
Edit Download
πŸ“„ template-loader.php File 3929
Edit Download
πŸ“„ template.php File 36834
Edit Download
πŸ“ theme-compat Folder -
πŸ“„ theme-i18n.json File 1526
Edit Download
πŸ“„ theme-previews.php File 2910
Edit Download
πŸ“„ theme-templates.php File 6238
Edit Download
πŸ“„ theme.json File 8921
Edit Download
πŸ“„ theme.php File 135008
Edit Download
πŸ“„ update.php File 38353
Edit Download
πŸ“„ user.php File 178062
Edit Download
πŸ“„ utf8.php File 7260
Edit Download
πŸ“„ vars.php File 6562
Edit Download
πŸ“„ version.php File 1104
Edit Download
πŸ“ widgets Folder -
πŸ“„ widgets.php File 71129
Edit Download
πŸ“„ wp-db.php File 445
Edit Download
πŸ“„ wp-diff.php File 799
Edit Download
'😯', * ':(' => 'πŸ™', * ':)' => 'πŸ™‚', * ':?' => 'πŸ˜•', * ) ); * * true === $smilies->contains( ':)' ); * false === $smilies->contains( 'simile' ); * * 'πŸ˜•' === $smilies->read_token( 'Not sure :?.', 9, $length_of_smily_syntax ); * 2 === $length_of_smily_syntax; * * ## Precomputing the Token Map. * * Creating the class involves some work sorting and organizing the tokens and their * replacement values. In order to skip this, it's possible for the class to export * its state and be used as actual PHP source code. * * Example: * * // Export with four spaces as the indent, only for the sake of this docblock. * // The default indent is a tab character. * $indent = ' '; * echo $smilies->precomputed_php_source_table( $indent ); * * // Output, to be pasted into a PHP source file: * WP_Token_Map::from_precomputed_table( * array( * "storage_version" => "6.6.0", * "key_length" => 2, * "groups" => "", * "long_words" => array(), * "small_words" => "8O\x00:)\x00:(\x00:?\x00", * "small_mappings" => array( "😯", "πŸ™‚", "πŸ™", "πŸ˜•" ) * ) * ); * * ## Large vs. small words. * * This class uses a short prefix called the "key" to optimize lookup of its tokens. * This means that some tokens may be shorter than or equal in length to that key. * Those words that are longer than the key are called "large" while those shorter * than or equal to the key length are called "small." * * This separation of large and small words is incidental to the way this class * optimizes lookup, and should be considered an internal implementation detail * of the class. It may still be important to be aware of it, however. * * ## Determining Key Length. * * The choice of the size of the key length should be based on the data being stored in * the token map. It should divide the data as evenly as possible, but should not create * so many groups that a large fraction of the groups only contain a single token. * * For the HTML5 named character references, a key length of 2 was found to provide a * sufficient spread and should be a good default for relatively large sets of tokens. * * However, for some data sets this might be too long. For example, a list of smilies * may be too small for a key length of 2. Perhaps 1 would be more appropriate. It's * best to experiment and determine empirically which values are appropriate. * * ## Generate Pre-Computed Source Code. * * Since the `WP_Token_Map` is designed for relatively static lookups, it can be * advantageous to precompute the values and instantiate a table that has already * sorted and grouped the tokens and built the lookup strings. * * This can be done with `WP_Token_Map::precomputed_php_source_table()`. * * Note that if there is a leading character that all tokens need, such as `&` for * HTML named character references, it can be beneficial to exclude this from the * token map. Instead, find occurrences of the leading character and then use the * token map to see if the following characters complete the token. * * Example: * * $map = WP_Token_Map::from_array( array( 'simple_smile:' => 'πŸ™‚', 'sob:' => '😭', 'soba:' => '🍜' ) ); * echo $map->precomputed_php_source_table(); * // Output * WP_Token_Map::from_precomputed_table( * array( * "storage_version" => "6.6.0", * "key_length" => 2, * "groups" => "si\x00so\x00", * "long_words" => array( * // simple_smile:[πŸ™‚]. * "\x0bmple_smile:\x04πŸ™‚", * // soba:[🍜] sob:[😭]. * "\x03ba:\x04🍜\x02b:\x04😭", * ), * "short_words" => "", * "short_mappings" => array() * } * ); * * This precomputed value can be stored directly in source code and will skip the * startup cost of generating the lookup strings. See `$html5_named_character_entities`. * * Note that any updates to the precomputed format should update the storage version * constant. It would also be best to provide an update function to take older known * versions and upgrade them in place when loading into `from_precomputed_table()`. * * ## Future Direction. * * It may be viable to dynamically increase the length limits such that there's no need to impose them. * The limit appears because of the packing structure, which indicates how many bytes each segment of * text in the lookup tables spans. If, however, care were taken to track the longest word length, then * the packing structure could change its representation to allow for that. Each additional byte storing * length, however, increases the memory overhead and lookup runtime. * * An alternative approach could be to borrow the UTF-8 variable-length encoding and store lengths of less * than 127 as a single byte with the high bit unset, storing longer lengths as the combination of * continuation bytes. * * Since it has not been shown during the development of this class that longer strings are required, this * update is deferred until such a need is clear. * * @since 6.6.0 */ class WP_Token_Map { /** * Denotes the version of the code which produces pre-computed source tables. * * This version will be used not only to verify pre-computed data, but also * to upgrade pre-computed data from older versions. Choosing a name that * corresponds to the WordPress release will help people identify where an * old copy of data came from. */ const STORAGE_VERSION = '6.6.0-trunk'; /** * Maximum length for each key and each transformed value in the table (in bytes). * * @since 6.6.0 */ const MAX_LENGTH = 256; /** * How many bytes of each key are used to form a group key for lookup. * This also determines whether a word is considered short or long. * * @since 6.6.0 * * @var int */ private $key_length = 2; /** * Stores an optimized form of the word set, where words are grouped * by a prefix of the `$key_length` and then collapsed into a string. * * In each group, the keys and lookups form a packed data structure. * The keys in the string are stripped of their "group key," which is * the prefix of length `$this->key_length` shared by all of the items * in the group. Each word in the string is prefixed by a single byte * whose raw unsigned integer value represents how many bytes follow. * * β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β” * β”‚ Length of rest β”‚ Rest of key β”‚ Length of value β”‚ Value β”‚ * β”‚ of key (bytes) β”‚ β”‚ (bytes) β”‚ β”‚ * β”œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”Όβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”Όβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”Όβ”€β”€β”€β”€β”€β”€β”€β”€β”€ * β”‚ 0x08 β”‚ nterDot; β”‚ 0x02 β”‚ Β· β”‚ * β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”΄β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”΄β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”΄β”€β”€β”€β”€β”€β”€β”€β”€β”˜ * * In this example, the key `CenterDot;` has a group key `Ce`, leaving * eight bytes for the rest of the key, `nterDot;`, and two bytes for * the transformed value `Β·` (or U+B7 or "\xC2\xB7"). * * Example: * * // Stores array( 'CenterDot;' => 'Β·', 'Cedilla;' => 'ΒΈ' ). * $groups = "Ce\x00"; * $large_words = array( "\x08nterDot;\x02Β·\x06dilla;\x02ΒΈ" ) * * The prefixes appear in the `$groups` string, each followed by a null * byte. This makes for quick lookup of where in the group string the key * is found, and then a simple division converts that offset into the index * in the `$large_words` array where the group string is to be found. * * This lookup data structure is designed to optimize cache locality and * minimize indirect memory reads when matching strings in the set. * * @since 6.6.0 * * @var array */ private $large_words = array(); /** * Stores the group keys for sequential string lookup. * * The offset into this string where the group key appears corresponds with the index * into the group array where the rest of the group string appears. This is an optimization * to improve cache locality while searching and minimize indirect memory accesses. * * @since 6.6.0 * * @var string */ private $groups = ''; /** * Stores an optimized row of small words, where every entry is * `$this->key_size + 1` bytes long and zero-extended. * * This packing allows for direct lookup of a short word followed * by the null byte, if extended to `$this->key_size + 1`. * * Example: * * // Stores array( 'GT', 'LT', 'gt', 'lt' ). * "GT\x00LT\x00gt\x00lt\x00" * * @since 6.6.0 * * @var string */ private $small_words = ''; /** * Replacements for the small words, in the same order they appear. * * With the position of a small word it's possible to index the translation * directly, as its position in the `$small_words` string corresponds to * the index of the replacement in the `$small_mapping` array. * * Example: * * array( '>', '<', '>', '<' ) * * @since 6.6.0 * * @var string[] */ private $small_mappings = array(); /** * Create a token map using an associative array of key/value pairs as the input. * * Example: * * $smilies = WP_Token_Map::from_array( array( * '8O' => '😯', * ':(' => 'πŸ™', * ':)' => 'πŸ™‚', * ':?' => 'πŸ˜•', * ) ); * * @since 6.6.0 * * @param array $mappings The keys transform into the values, both are strings. * @param int $key_length Determines the group key length. Leave at the default value * of 2 unless there's an empirical reason to change it. * * @return WP_Token_Map|null Token map, unless unable to create it. */ public static function from_array( array $mappings, int $key_length = 2 ): ?WP_Token_Map { $map = new WP_Token_Map(); $map->key_length = $key_length; // Start by grouping words. $groups = array(); $shorts = array(); foreach ( $mappings as $word => $mapping ) { if ( self::MAX_LENGTH <= strlen( $word ) || self::MAX_LENGTH <= strlen( $mapping ) ) { _doing_it_wrong( __METHOD__, sprintf( /* translators: 1: maximum byte length (a count) */ __( 'Token Map tokens and substitutions must all be shorter than %1$d bytes.' ), self::MAX_LENGTH ), '6.6.0' ); return null; } $length = strlen( $word ); if ( $key_length >= $length ) { $shorts[] = $word; } else { $group = substr( $word, 0, $key_length ); if ( ! isset( $groups[ $group ] ) ) { $groups[ $group ] = array(); } $groups[ $group ][] = array( substr( $word, $key_length ), $mapping ); } } /* * Sort the words to ensure that no smaller substring of a match masks the full match. * For example, `Cap` should not match before `CapitalDifferentialD`. */ usort( $shorts, 'WP_Token_Map::longest_first_then_alphabetical' ); foreach ( $groups as $group_key => $group ) { usort( $groups[ $group_key ], static function ( array $a, array $b ): int { return self::longest_first_then_alphabetical( $a[0], $b[0] ); } ); } // Finally construct the optimized lookups. foreach ( $shorts as $word ) { $map->small_words .= str_pad( $word, $key_length + 1, "\x00", STR_PAD_RIGHT ); $map->small_mappings[] = $mappings[ $word ]; } $group_keys = array_keys( $groups ); sort( $group_keys ); foreach ( $group_keys as $group ) { $map->groups .= "{$group}\x00"; $group_string = ''; foreach ( $groups[ $group ] as $group_word ) { list( $word, $mapping ) = $group_word; $word_length = pack( 'C', strlen( $word ) ); $mapping_length = pack( 'C', strlen( $mapping ) ); $group_string .= "{$word_length}{$word}{$mapping_length}{$mapping}"; } $map->large_words[] = $group_string; } return $map; } /** * Creates a token map from a pre-computed table. * This skips the initialization cost of generating the table. * * This function should only be used to load data created with * WP_Token_Map::precomputed_php_source_tag(). * * @since 6.6.0 * * @param array $state { * Stores pre-computed state for directly loading into a Token Map. * * @type string $storage_version Which version of the code produced this state. * @type int $key_length Group key length. * @type string $groups Group lookup index. * @type array $large_words Large word groups and packed strings. * @type string $small_words Small words packed string. * @type array $small_mappings Small word mappings. * } * * @return WP_Token_Map Map with precomputed data loaded. */ public static function from_precomputed_table( $state ): ?WP_Token_Map { $has_necessary_state = isset( $state['storage_version'], $state['key_length'], $state['groups'], $state['large_words'], $state['small_words'], $state['small_mappings'] ); if ( ! $has_necessary_state ) { _doing_it_wrong( __METHOD__, __( 'Missing required inputs to pre-computed WP_Token_Map.' ), '6.6.0' ); return null; } if ( self::STORAGE_VERSION !== $state['storage_version'] ) { _doing_it_wrong( __METHOD__, /* translators: 1: version string, 2: version string. */ sprintf( __( 'Loaded version \'%1$s\' incompatible with expected version \'%2$s\'.' ), $state['storage_version'], self::STORAGE_VERSION ), '6.6.0' ); return null; } $map = new WP_Token_Map(); $map->key_length = $state['key_length']; $map->groups = $state['groups']; $map->large_words = $state['large_words']; $map->small_words = $state['small_words']; $map->small_mappings = $state['small_mappings']; return $map; } /** * Indicates if a given word is a lookup key in the map. * * Example: * * true === $smilies->contains( ':)' ); * false === $smilies->contains( 'simile' ); * * @since 6.6.0 * * @param string $word Determine if this word is a lookup key in the map. * @param string $case_sensitivity Optional. Pass 'ascii-case-insensitive' to ignore ASCII case when matching. Default 'case-sensitive'. * @return bool Whether there's an entry for the given word in the map. */ public function contains( string $word, string $case_sensitivity = 'case-sensitive' ): bool { $ignore_case = 'ascii-case-insensitive' === $case_sensitivity; if ( $this->key_length >= strlen( $word ) ) { if ( 0 === strlen( $this->small_words ) ) { return false; } $term = str_pad( $word, $this->key_length + 1, "\x00", STR_PAD_RIGHT ); $word_at = $ignore_case ? stripos( $this->small_words, $term ) : strpos( $this->small_words, $term ); if ( false === $word_at ) { return false; } return true; } $group_key = substr( $word, 0, $this->key_length ); $group_at = $ignore_case ? stripos( $this->groups, $group_key ) : strpos( $this->groups, $group_key ); if ( false === $group_at ) { return false; } $group = $this->large_words[ $group_at / ( $this->key_length + 1 ) ]; $group_length = strlen( $group ); $slug = substr( $word, $this->key_length ); $length = strlen( $slug ); $at = 0; while ( $at < $group_length ) { $token_length = unpack( 'C', $group[ $at++ ] )[1]; $token_at = $at; $at += $token_length; $mapping_length = unpack( 'C', $group[ $at++ ] )[1]; $mapping_at = $at; if ( $token_length === $length && 0 === substr_compare( $group, $slug, $token_at, $token_length, $ignore_case ) ) { return true; } $at = $mapping_at + $mapping_length; } return false; } /** * If the text starting at a given offset is a lookup key in the map, * return the corresponding transformation from the map, else `false`. * * This function returns the translated string, but accepts an optional * parameter `$matched_token_byte_length`, which communicates how many * bytes long the lookup key was, if it found one. This can be used to * advance a cursor in calling code if a lookup key was found. * * Example: * * false === $smilies->read_token( 'Not sure :?.', 0, $token_byte_length ); * 'πŸ˜•' === $smilies->read_token( 'Not sure :?.', 9, $token_byte_length ); * 2 === $token_byte_length; * * Example: * * while ( $at < strlen( $input ) ) { * $next_at = strpos( $input, ':', $at ); * if ( false === $next_at ) { * break; * } * * $smily = $smilies->read_token( $input, $next_at, $token_byte_length ); * if ( false === $next_at ) { * ++$at; * continue; * } * * $prefix = substr( $input, $at, $next_at - $at ); * $at += $token_byte_length; * $output .= "{$prefix}{$smily}"; * } * * @since 6.6.0 * * @param string $text String in which to search for a lookup key. * @param int $offset Optional. How many bytes into the string where the lookup key ought to start. Default 0. * @param int|null &$matched_token_byte_length Optional. Holds byte-length of found token matched, otherwise not set. Default null. * @param string $case_sensitivity Optional. Pass 'ascii-case-insensitive' to ignore ASCII case when matching. Default 'case-sensitive'. * * @return string|null Mapped value of lookup key if found, otherwise `null`. */ public function read_token( string $text, int $offset = 0, &$matched_token_byte_length = null, $case_sensitivity = 'case-sensitive' ): ?string { $ignore_case = 'ascii-case-insensitive' === $case_sensitivity; $text_length = strlen( $text ); // Search for a long word first, if the text is long enough, and if that fails, a short one. if ( $text_length > $this->key_length ) { $group_key = substr( $text, $offset, $this->key_length ); $group_at = $ignore_case ? stripos( $this->groups, $group_key ) : strpos( $this->groups, $group_key ); if ( false === $group_at ) { // Perhaps a short word then. return strlen( $this->small_words ) > 0 ? $this->read_small_token( $text, $offset, $matched_token_byte_length, $case_sensitivity ) : null; } $group = $this->large_words[ $group_at / ( $this->key_length + 1 ) ]; $group_length = strlen( $group ); $at = 0; while ( $at < $group_length ) { $token_length = unpack( 'C', $group[ $at++ ] )[1]; $token = substr( $group, $at, $token_length ); $at += $token_length; $mapping_length = unpack( 'C', $group[ $at++ ] )[1]; $mapping_at = $at; if ( 0 === substr_compare( $text, $token, $offset + $this->key_length, $token_length, $ignore_case ) ) { $matched_token_byte_length = $this->key_length + $token_length; return substr( $group, $mapping_at, $mapping_length ); } $at = $mapping_at + $mapping_length; } } // Perhaps a short word then. return strlen( $this->small_words ) > 0 ? $this->read_small_token( $text, $offset, $matched_token_byte_length, $case_sensitivity ) : null; } /** * Finds a match for a short word at the index. * * @since 6.6.0 * * @param string $text String in which to search for a lookup key. * @param int $offset Optional. How many bytes into the string where the lookup key ought to start. Default 0. * @param int|null &$matched_token_byte_length Optional. Holds byte-length of found lookup key if matched, otherwise not set. Default null. * @param string $case_sensitivity Optional. Pass 'ascii-case-insensitive' to ignore ASCII case when matching. Default 'case-sensitive'. * * @return string|null Mapped value of lookup key if found, otherwise `null`. */ private function read_small_token( string $text, int $offset = 0, &$matched_token_byte_length = null, $case_sensitivity = 'case-sensitive' ): ?string { $ignore_case = 'ascii-case-insensitive' === $case_sensitivity; $small_length = strlen( $this->small_words ); $search_text = substr( $text, $offset, $this->key_length ); if ( $ignore_case ) { $search_text = strtoupper( $search_text ); } $starting_char = $search_text[0]; $at = 0; while ( $at < $small_length ) { if ( $starting_char !== $this->small_words[ $at ] && ( ! $ignore_case || strtoupper( $this->small_words[ $at ] ) !== $starting_char ) ) { $at += $this->key_length + 1; continue; } for ( $adjust = 1; $adjust < $this->key_length; $adjust++ ) { if ( "\x00" === $this->small_words[ $at + $adjust ] ) { $matched_token_byte_length = $adjust; return $this->small_mappings[ $at / ( $this->key_length + 1 ) ]; } if ( $search_text[ $adjust ] !== $this->small_words[ $at + $adjust ] && ( ! $ignore_case || strtoupper( $this->small_words[ $at + $adjust ] !== $search_text[ $adjust ] ) ) ) { $at += $this->key_length + 1; continue 2; } } $matched_token_byte_length = $adjust; return $this->small_mappings[ $at / ( $this->key_length + 1 ) ]; } return null; } /** * Exports the token map into an associate array of key/value pairs. * * Example: * * $smilies->to_array() === array( * '8O' => '😯', * ':(' => 'πŸ™', * ':)' => 'πŸ™‚', * ':?' => 'πŸ˜•', * ); * * @return array The lookup key/substitution values as an associate array. */ public function to_array(): array { $tokens = array(); $at = 0; $small_mapping = 0; $small_length = strlen( $this->small_words ); while ( $at < $small_length ) { $key = rtrim( substr( $this->small_words, $at, $this->key_length + 1 ), "\x00" ); $value = $this->small_mappings[ $small_mapping++ ]; $tokens[ $key ] = $value; $at += $this->key_length + 1; } foreach ( $this->large_words as $index => $group ) { $prefix = substr( $this->groups, $index * ( $this->key_length + 1 ), 2 ); $group_length = strlen( $group ); $at = 0; while ( $at < $group_length ) { $length = unpack( 'C', $group[ $at++ ] )[1]; $key = $prefix . substr( $group, $at, $length ); $at += $length; $length = unpack( 'C', $group[ $at++ ] )[1]; $value = substr( $group, $at, $length ); $tokens[ $key ] = $value; $at += $length; } } return $tokens; } /** * Export the token map for quick loading in PHP source code. * * This function has a specific purpose, to make loading of static token maps fast. * It's used to ensure that the HTML character reference lookups add a minimal cost * to initializing the PHP process. * * Example: * * echo $smilies->precomputed_php_source_table(); * * // Output. * WP_Token_Map::from_precomputed_table( * array( * "storage_version" => "6.6.0", * "key_length" => 2, * "groups" => "", * "long_words" => array(), * "small_words" => "8O\x00:)\x00:(\x00:?\x00", * "small_mappings" => array( "😯", "πŸ™‚", "πŸ™", "πŸ˜•" ) * ) * ); * * @since 6.6.0 * * @param string $indent Optional. Use this string for indentation, or rely on the default horizontal tab character. Default "\t". * @return string Value which can be pasted into a PHP source file for quick loading of table. */ public function precomputed_php_source_table( string $indent = "\t" ): string { $i1 = $indent; $i2 = $i1 . $indent; $i3 = $i2 . $indent; $class_version = self::STORAGE_VERSION; $output = self::class . "::from_precomputed_table(\n"; $output .= "{$i1}array(\n"; $output .= "{$i2}\"storage_version\" => \"{$class_version}\",\n"; $output .= "{$i2}\"key_length\" => {$this->key_length},\n"; $group_line = str_replace( "\x00", "\\x00", $this->groups ); $output .= "{$i2}\"groups\" => \"{$group_line}\",\n"; $output .= "{$i2}\"large_words\" => array(\n"; $prefixes = explode( "\x00", $this->groups ); foreach ( $prefixes as $index => $prefix ) { if ( '' === $prefix ) { break; } $group = $this->large_words[ $index ]; $group_length = strlen( $group ); $comment_line = "{$i3}//"; $data_line = "{$i3}\""; $at = 0; while ( $at < $group_length ) { $token_length = unpack( 'C', $group[ $at++ ] )[1]; $token = substr( $group, $at, $token_length ); $at += $token_length; $mapping_length = unpack( 'C', $group[ $at++ ] )[1]; $mapping = substr( $group, $at, $mapping_length ); $at += $mapping_length; $token_digits = str_pad( dechex( $token_length ), 2, '0', STR_PAD_LEFT ); $mapping_digits = str_pad( dechex( $mapping_length ), 2, '0', STR_PAD_LEFT ); $mapping = preg_replace_callback( "~[\\x00-\\x1f\\x22\\x5c]~", static function ( $match_result ) { switch ( $match_result[0] ) { case '"': return '\\"'; case '\\': return '\\\\'; default: $hex = dechex( ord( $match_result[0] ) ); return "\\x{$hex}"; } }, $mapping ); $comment_line .= " {$prefix}{$token}[{$mapping}]"; $data_line .= "\\x{$token_digits}{$token}\\x{$mapping_digits}{$mapping}"; } $comment_line .= ".\n"; $data_line .= "\",\n"; $output .= $comment_line; $output .= $data_line; } $output .= "{$i2}),\n"; $small_words = array(); $small_length = strlen( $this->small_words ); $at = 0; while ( $at < $small_length ) { $small_words[] = substr( $this->small_words, $at, $this->key_length + 1 ); $at += $this->key_length + 1; } $small_text = str_replace( "\x00", '\x00', implode( '', $small_words ) ); $output .= "{$i2}\"small_words\" => \"{$small_text}\",\n"; $output .= "{$i2}\"small_mappings\" => array(\n"; foreach ( $this->small_mappings as $mapping ) { $output .= "{$i3}\"{$mapping}\",\n"; } $output .= "{$i2})\n"; $output .= "{$i1})\n"; $output .= ')'; return $output; } /** * Compares two strings, returning the longest, or whichever * is first alphabetically if they are the same length. * * This is an important sort when building the token map because * it should not form a match on a substring of a longer potential * match. For example, it should not detect `Cap` when matching * against the string `CapitalDifferentialD`. * * @since 6.6.0 * * @param string $a First string to compare. * @param string $b Second string to compare. * @return int -1 or lower if `$a` is less than `$b`; 1 or greater if `$a` is greater than `$b`, and 0 if they are equal. */ private static function longest_first_then_alphabetical( string $a, string $b ): int { if ( $a === $b ) { return 0; } $length_a = strlen( $a ); $length_b = strlen( $b ); // Longer strings are less-than for comparison's sake. if ( $length_a !== $length_b ) { return $length_b - $length_a; } return strcmp( $a, $b ); } }